1use serde::{Deserialize, Serialize};
31
32pub const TYPE_APPROVAL_USE: &str = "treeship/approval-use/v1";
37pub const TYPE_APPROVAL_REVOCATION: &str = "treeship/approval-revocation/v1";
38pub const TYPE_JOURNAL_CHECKPOINT: &str = "treeship/journal-checkpoint/v1";
39
40#[derive(Debug, Clone, Serialize, Deserialize)]
61pub struct ApprovalUse {
62 #[serde(rename = "type")]
63 pub type_: String,
64
65 pub use_id: String,
69
70 pub grant_id: String,
72
73 pub grant_digest: String,
77
78 pub nonce_digest: String,
83
84 pub actor: String,
85 pub action: String,
86 pub subject: String,
90
91 #[serde(default, skip_serializing_if = "Option::is_none")]
94 pub session_id: Option<String>,
95
96 #[serde(default, skip_serializing_if = "Option::is_none")]
100 pub action_artifact_id: Option<String>,
101
102 #[serde(default, skip_serializing_if = "Option::is_none")]
104 pub receipt_digest: Option<String>,
105
106 pub use_number: u32,
109
110 #[serde(default, skip_serializing_if = "Option::is_none")]
114 pub max_uses: Option<u32>,
115
116 #[serde(default, skip_serializing_if = "Option::is_none")]
121 pub idempotency_key: Option<String>,
122
123 pub created_at: String,
124
125 #[serde(default, skip_serializing_if = "Option::is_none")]
129 pub expires_at: Option<String>,
130
131 #[serde(default)]
134 pub previous_record_digest: String,
135
136 #[serde(default)]
139 pub record_digest: String,
140
141 #[serde(default, skip_serializing_if = "Option::is_none")]
145 pub signature: Option<String>,
146 #[serde(default, skip_serializing_if = "Option::is_none")]
147 pub signature_alg: Option<String>,
148 #[serde(default, skip_serializing_if = "Option::is_none")]
149 pub signing_key_id: Option<String>,
150}
151
152#[derive(Debug, Clone, Serialize, Deserialize)]
164pub struct ApprovalRevocation {
165 #[serde(rename = "type")]
166 pub type_: String,
167 pub revocation_id: String,
168 pub grant_id: String,
169 pub grant_digest: String,
170 pub revoker: String,
171 pub reason: Option<String>,
172 pub created_at: String,
173 #[serde(default)]
174 pub previous_record_digest: String,
175 #[serde(default)]
176 pub record_digest: String,
177 #[serde(default, skip_serializing_if = "Option::is_none")]
178 pub signature: Option<String>,
179 #[serde(default, skip_serializing_if = "Option::is_none")]
180 pub signature_alg: Option<String>,
181 #[serde(default, skip_serializing_if = "Option::is_none")]
182 pub signing_key_id: Option<String>,
183}
184
185#[derive(Debug, Clone, Copy, PartialEq, Eq, Default, Serialize, Deserialize)]
205#[serde(rename_all = "kebab-case")]
206pub enum CheckpointKind {
207 #[default]
210 LocalJournal,
211 HubOrg,
214}
215
216impl CheckpointKind {
217 pub fn label(self) -> &'static str {
218 match self {
219 Self::LocalJournal => "local-journal",
220 Self::HubOrg => "hub-org",
221 }
222 }
223}
224
225#[derive(Debug, Clone, Serialize, Deserialize)]
240pub struct JournalCheckpoint {
241 #[serde(rename = "type")]
242 pub type_: String,
243 pub checkpoint_id: String,
244
245 #[serde(default)]
248 pub checkpoint_kind: CheckpointKind,
249
250 pub from_record_index: u64,
253 pub to_record_index: u64,
254
255 pub merkle_root: String,
258 pub leaf_count: u64,
259
260 pub journal_id: String,
261 pub created_at: String,
262
263 #[serde(default, skip_serializing_if = "String::is_empty")]
266 pub hub_id: String,
267
268 #[serde(default, skip_serializing_if = "String::is_empty")]
273 pub hub_public_key: String,
274
275 #[serde(default, skip_serializing_if = "String::is_empty")]
279 pub hub_signature: String,
280
281 #[serde(default, skip_serializing_if = "String::is_empty")]
285 pub signed_at: String,
286
287 #[serde(default, skip_serializing_if = "Vec::is_empty")]
291 pub covered_use_ids: Vec<String>,
292
293 #[serde(default, skip_serializing_if = "Vec::is_empty")]
296 pub covered_grant_ids: Vec<String>,
297
298 #[serde(default)]
299 pub previous_record_digest: String,
300 #[serde(default)]
301 pub record_digest: String,
302
303 #[serde(default, skip_serializing_if = "Option::is_none")]
304 pub signature: Option<String>,
305 #[serde(default, skip_serializing_if = "Option::is_none")]
306 pub signature_alg: Option<String>,
307 #[serde(default, skip_serializing_if = "Option::is_none")]
308 pub signing_key_id: Option<String>,
309}
310
311impl JournalCheckpoint {
312 pub fn is_hub_signed(&self) -> bool {
316 self.checkpoint_kind == CheckpointKind::HubOrg
317 && !self.hub_id.is_empty()
318 && !self.hub_public_key.is_empty()
319 && !self.hub_signature.is_empty()
320 && !self.signed_at.is_empty()
321 }
322
323 pub fn canonical_hub_signing_bytes(&self) -> Vec<u8> {
328 #[derive(Serialize)]
333 struct Signing<'a> {
334 #[serde(rename = "type")] type_: &'a str,
335 checkpoint_id: &'a str,
336 checkpoint_kind: CheckpointKind,
337 from_record_index: u64,
338 to_record_index: u64,
339 merkle_root: &'a str,
340 leaf_count: u64,
341 journal_id: &'a str,
342 created_at: &'a str,
343 hub_id: &'a str,
344 hub_public_key: &'a str,
345 signed_at: &'a str,
346 covered_use_ids: &'a [String],
347 covered_grant_ids: &'a [String],
348 previous_record_digest: &'a str,
349 }
350 let v = Signing {
351 type_: &self.type_,
352 checkpoint_id: &self.checkpoint_id,
353 checkpoint_kind: self.checkpoint_kind,
354 from_record_index: self.from_record_index,
355 to_record_index: self.to_record_index,
356 merkle_root: &self.merkle_root,
357 leaf_count: self.leaf_count,
358 journal_id: &self.journal_id,
359 created_at: &self.created_at,
360 hub_id: &self.hub_id,
361 hub_public_key: &self.hub_public_key,
362 signed_at: &self.signed_at,
363 covered_use_ids: &self.covered_use_ids,
364 covered_grant_ids: &self.covered_grant_ids,
365 previous_record_digest: &self.previous_record_digest,
366 };
367 serde_json::to_vec(&v).unwrap_or_default()
368 }
369}
370
371#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize)]
383#[serde(rename_all = "kebab-case")]
384pub enum ReplayCheckLevel {
385 NotPerformed,
387 PackageLocal,
390 LocalJournal,
392 HubOrg,
394}
395
396impl ReplayCheckLevel {
397 pub fn label(self) -> &'static str {
398 match self {
399 Self::NotPerformed => "not performed",
400 Self::PackageLocal => "package-local",
401 Self::LocalJournal => "local-journal",
402 Self::HubOrg => "hub-org",
403 }
404 }
405}
406
407#[derive(Debug, Clone, Serialize, Deserialize)]
411pub struct ReplayCheck {
412 pub level: ReplayCheckLevel,
413
414 #[serde(default, skip_serializing_if = "Option::is_none")]
417 pub use_number: Option<u32>,
418
419 #[serde(default, skip_serializing_if = "Option::is_none")]
421 pub max_uses: Option<u32>,
422
423 #[serde(default, skip_serializing_if = "Option::is_none")]
427 pub passed: Option<bool>,
428
429 #[serde(default, skip_serializing_if = "Option::is_none")]
431 pub details: Option<String>,
432}
433
434impl ReplayCheck {
435 pub fn not_performed() -> Self {
436 Self { level: ReplayCheckLevel::NotPerformed, use_number: None, max_uses: None, passed: None, details: None }
437 }
438
439 pub fn package_local(passed: bool, details: impl Into<String>) -> Self {
440 Self {
441 level: ReplayCheckLevel::PackageLocal,
442 use_number: None,
443 max_uses: None,
444 passed: Some(passed),
445 details: Some(details.into()),
446 }
447 }
448}
449
450pub fn approval_use_record_digest(rec: &ApprovalUse) -> String {
463 use sha2::{Digest, Sha256};
464 let mut canon = rec.clone();
465 canon.record_digest = String::new();
466 let bytes = serde_json::to_vec(&canon).unwrap_or_default();
467 let mut hasher = Sha256::new();
468 hasher.update(&bytes);
469 let digest = hasher.finalize();
470 let mut hex = String::with_capacity(64 + 7);
471 hex.push_str("sha256:");
472 for b in digest.as_slice() {
473 use std::fmt::Write;
474 let _ = write!(hex, "{b:02x}");
475 }
476 hex
477}
478
479pub fn approval_revocation_record_digest(rec: &ApprovalRevocation) -> String {
480 use sha2::{Digest, Sha256};
481 let mut canon = rec.clone();
482 canon.record_digest = String::new();
483 let bytes = serde_json::to_vec(&canon).unwrap_or_default();
484 let mut hasher = Sha256::new();
485 hasher.update(&bytes);
486 let digest = hasher.finalize();
487 let mut hex = String::with_capacity(64 + 7);
488 hex.push_str("sha256:");
489 for b in digest.as_slice() {
490 use std::fmt::Write;
491 let _ = write!(hex, "{b:02x}");
492 }
493 hex
494}
495
496pub fn journal_checkpoint_record_digest(rec: &JournalCheckpoint) -> String {
497 use sha2::{Digest, Sha256};
498 let mut canon = rec.clone();
499 canon.record_digest = String::new();
500 let bytes = serde_json::to_vec(&canon).unwrap_or_default();
501 let mut hasher = Sha256::new();
502 hasher.update(&bytes);
503 let digest = hasher.finalize();
504 let mut hex = String::with_capacity(64 + 7);
505 hex.push_str("sha256:");
506 for b in digest.as_slice() {
507 use std::fmt::Write;
508 let _ = write!(hex, "{b:02x}");
509 }
510 hex
511}
512
513#[derive(Debug, Clone, PartialEq, Eq)]
515pub enum HubCheckpointVerification {
516 Valid,
519 MissingFields(&'static str),
524 Tampered,
527 NotHubKind,
531}
532
533pub fn verify_hub_checkpoint_signature(
545 cp: &JournalCheckpoint,
546) -> HubCheckpointVerification {
547 use base64::{engine::general_purpose::URL_SAFE_NO_PAD, Engine};
548 use ed25519_dalek::{Signature, Verifier, VerifyingKey};
549
550 if cp.checkpoint_kind != CheckpointKind::HubOrg {
551 return HubCheckpointVerification::NotHubKind;
552 }
553 if cp.hub_id.is_empty() { return HubCheckpointVerification::MissingFields("hub_id"); }
554 if cp.hub_public_key.is_empty() { return HubCheckpointVerification::MissingFields("hub_public_key"); }
555 if cp.hub_signature.is_empty() { return HubCheckpointVerification::MissingFields("hub_signature"); }
556 if cp.signed_at.is_empty() { return HubCheckpointVerification::MissingFields("signed_at"); }
557
558 let pk_bytes = match URL_SAFE_NO_PAD.decode(cp.hub_public_key.as_bytes()) {
559 Ok(b) if b.len() == 32 => b,
560 _ => return HubCheckpointVerification::Tampered,
561 };
562 let sig_bytes = match URL_SAFE_NO_PAD.decode(cp.hub_signature.as_bytes()) {
563 Ok(b) if b.len() == 64 => b,
564 _ => return HubCheckpointVerification::Tampered,
565 };
566 let mut pk_arr = [0u8; 32];
567 pk_arr.copy_from_slice(&pk_bytes);
568 let mut sig_arr = [0u8; 64];
569 sig_arr.copy_from_slice(&sig_bytes);
570
571 let vk = match VerifyingKey::from_bytes(&pk_arr) {
572 Ok(k) => k,
573 Err(_) => return HubCheckpointVerification::Tampered,
574 };
575 let sig = Signature::from_bytes(&sig_arr);
576 let payload = cp.canonical_hub_signing_bytes();
577 match vk.verify(&payload, &sig) {
578 Ok(()) => HubCheckpointVerification::Valid,
579 Err(_) => HubCheckpointVerification::Tampered,
580 }
581}
582
583pub fn nonce_digest(raw_nonce: &str) -> String {
586 use sha2::{Digest, Sha256};
587 let mut hasher = Sha256::new();
588 hasher.update(raw_nonce.as_bytes());
589 let digest = hasher.finalize();
590 let mut hex = String::with_capacity(64 + 7);
591 hex.push_str("sha256:");
592 for b in digest.as_slice() {
593 use std::fmt::Write;
594 let _ = write!(hex, "{b:02x}");
595 }
596 hex
597}
598
599#[cfg(test)]
604mod tests {
605 use super::*;
606
607 fn sample_use() -> ApprovalUse {
608 ApprovalUse {
609 type_: TYPE_APPROVAL_USE.into(),
610 use_id: "use_abc".into(),
611 grant_id: "art_grant_1".into(),
612 grant_digest: "sha256:00".into(),
613 nonce_digest: "sha256:11".into(),
614 actor: "agent://deployer".into(),
615 action: "deploy.production".into(),
616 subject: "env://production".into(),
617 session_id: Some("ssn_xyz".into()),
618 action_artifact_id: None,
619 receipt_digest: None,
620 use_number: 1,
621 max_uses: Some(1),
622 idempotency_key: None,
623 created_at: "2026-04-30T06:00:00Z".into(),
624 expires_at: None,
625 previous_record_digest: String::new(),
626 record_digest: String::new(),
627 signature: None,
628 signature_alg: None,
629 signing_key_id: None,
630 }
631 }
632
633 #[test]
634 fn approval_use_serialization_round_trips() {
635 let u = sample_use();
636 let bytes = serde_json::to_vec(&u).unwrap();
637 let back: ApprovalUse = serde_json::from_slice(&bytes).unwrap();
638 assert_eq!(back.use_id, u.use_id);
639 assert_eq!(back.grant_id, u.grant_id);
640 assert_eq!(back.use_number, 1);
641 }
642
643 #[test]
644 fn record_digest_is_stable_and_excludes_itself() {
645 let u1 = sample_use();
649 let mut u2 = u1.clone();
650 u2.record_digest = "sha256:cafe".into();
651 assert_eq!(approval_use_record_digest(&u1), approval_use_record_digest(&u2));
652 }
653
654 #[test]
655 fn previous_record_digest_chains() {
656 let mut a = sample_use();
660 a.use_number = 1;
661 a.record_digest = approval_use_record_digest(&a);
662
663 let mut b = sample_use();
664 b.use_number = 2;
665 b.use_id = "use_def".into();
666 b.previous_record_digest = a.record_digest.clone();
667 b.record_digest = approval_use_record_digest(&b);
668
669 assert_eq!(b.previous_record_digest, a.record_digest);
670 let mut c = sample_use();
672 c.use_id = "use_ghi".into();
673 c.use_number = 2;
674 c.previous_record_digest = "sha256:wrong".into();
675 c.record_digest = approval_use_record_digest(&c);
676 assert_ne!(b.record_digest, c.record_digest);
677 }
678
679 #[test]
680 fn nonce_digest_does_not_leak_raw_nonce() {
681 let raw = "n_abcdef0123";
685 let d = nonce_digest(raw);
686 assert!(d.starts_with("sha256:"));
687 assert!(!d.contains(raw), "digest must not contain the raw nonce");
688 }
689
690 #[test]
691 fn replay_check_level_labels() {
692 assert_eq!(ReplayCheckLevel::NotPerformed.label(), "not performed");
693 assert_eq!(ReplayCheckLevel::PackageLocal.label(), "package-local");
694 assert_eq!(ReplayCheckLevel::LocalJournal.label(), "local-journal");
695 assert_eq!(ReplayCheckLevel::HubOrg.label(), "hub-org");
696 }
697
698 #[test]
699 fn replay_check_serialization_uses_kebab_case() {
700 let r = ReplayCheck {
701 level: ReplayCheckLevel::LocalJournal,
702 use_number: Some(1),
703 max_uses: Some(1),
704 passed: Some(true),
705 details: Some("local Approval Use Journal passed".into()),
706 };
707 let v = serde_json::to_value(&r).unwrap();
708 assert_eq!(v["level"], "local-journal");
709 assert_eq!(v["use_number"], 1);
710 assert_eq!(v["max_uses"], 1);
711 assert_eq!(v["passed"], true);
712 }
713
714 #[test]
715 fn revocation_record_digest_stable() {
716 let rev = ApprovalRevocation {
717 type_: TYPE_APPROVAL_REVOCATION.into(),
718 revocation_id: "rev_1".into(),
719 grant_id: "art_grant_1".into(),
720 grant_digest: "sha256:00".into(),
721 revoker: "human://alice".into(),
722 reason: Some("rotated key".into()),
723 created_at: "2026-04-30T06:01:00Z".into(),
724 previous_record_digest: "sha256:00".into(),
725 record_digest: String::new(),
726 signature: None,
727 signature_alg: None,
728 signing_key_id: None,
729 };
730 let d1 = approval_revocation_record_digest(&rev);
731 let d2 = approval_revocation_record_digest(&rev);
732 assert_eq!(d1, d2);
733 }
734
735 fn sample_checkpoint(kind: CheckpointKind) -> JournalCheckpoint {
736 JournalCheckpoint {
737 type_: TYPE_JOURNAL_CHECKPOINT.into(),
738 checkpoint_id: "cp_1".into(),
739 checkpoint_kind: kind,
740 from_record_index: 1,
741 to_record_index: 10,
742 merkle_root: "sha256:abcd".into(),
743 leaf_count: 10,
744 journal_id: "journal_1".into(),
745 created_at: "2026-04-30T06:02:00Z".into(),
746 hub_id: String::new(),
747 hub_public_key: String::new(),
748 hub_signature: String::new(),
749 signed_at: String::new(),
750 covered_use_ids: Vec::new(),
751 covered_grant_ids: Vec::new(),
752 previous_record_digest: "sha256:00".into(),
753 record_digest: String::new(),
754 signature: None,
755 signature_alg: None,
756 signing_key_id: None,
757 }
758 }
759
760 #[test]
761 fn checkpoint_record_digest_stable() {
762 let cp = sample_checkpoint(CheckpointKind::LocalJournal);
763 let d1 = journal_checkpoint_record_digest(&cp);
764 let d2 = journal_checkpoint_record_digest(&cp);
765 assert_eq!(d1, d2);
766 }
767
768 #[test]
769 fn checkpoint_kind_defaults_to_local_journal() {
770 let json = r#"{"type":"treeship/journal-checkpoint/v1","checkpoint_id":"cp_legacy",
774 "from_record_index":1,"to_record_index":10,"merkle_root":"sha256:00",
775 "leaf_count":10,"journal_id":"j","created_at":"2026-04-30T00:00:00Z"}"#;
776 let cp: JournalCheckpoint = serde_json::from_str(json).unwrap();
777 assert_eq!(cp.checkpoint_kind, CheckpointKind::LocalJournal);
778 assert!(!cp.is_hub_signed());
779 }
780
781 #[test]
782 fn checkpoint_kind_serializes_kebab_case() {
783 let cp = sample_checkpoint(CheckpointKind::HubOrg);
784 let v = serde_json::to_value(&cp).unwrap();
785 assert_eq!(v["checkpoint_kind"], "hub-org");
786 }
787
788 #[test]
789 fn local_journal_checkpoint_is_not_hub_signed() {
790 let cp = sample_checkpoint(CheckpointKind::LocalJournal);
791 assert!(!cp.is_hub_signed());
792 assert_eq!(
793 verify_hub_checkpoint_signature(&cp),
794 HubCheckpointVerification::NotHubKind,
795 );
796 }
797
798 #[test]
799 fn hub_kind_without_fields_is_missing() {
800 let cp = sample_checkpoint(CheckpointKind::HubOrg);
801 assert!(!cp.is_hub_signed());
802 assert!(matches!(
803 verify_hub_checkpoint_signature(&cp),
804 HubCheckpointVerification::MissingFields(_),
805 ));
806 }
807
808 #[test]
813 fn hub_checkpoint_signature_round_trip() {
814 use base64::{engine::general_purpose::URL_SAFE_NO_PAD, Engine};
815 use ed25519_dalek::{Signer, SigningKey};
816
817 let mut sk_bytes = [0u8; 32];
818 for (i, b) in sk_bytes.iter_mut().enumerate() {
819 *b = i as u8 + 7;
820 }
821 let sk = SigningKey::from_bytes(&sk_bytes);
822 let pk = sk.verifying_key();
823 let pk_b64 = URL_SAFE_NO_PAD.encode(pk.to_bytes());
824
825 let mut cp = sample_checkpoint(CheckpointKind::HubOrg);
826 cp.hub_id = "hub://zerker-org".into();
827 cp.hub_public_key = pk_b64.clone();
828 cp.signed_at = "2026-04-30T07:00:00Z".into();
829 cp.covered_use_ids = vec!["use_alpha".into(), "use_beta".into()];
830
831 let payload = cp.canonical_hub_signing_bytes();
832 let sig = sk.sign(&payload);
833 cp.hub_signature = URL_SAFE_NO_PAD.encode(sig.to_bytes());
834
835 assert!(cp.is_hub_signed());
836 assert_eq!(
837 verify_hub_checkpoint_signature(&cp),
838 HubCheckpointVerification::Valid,
839 );
840 }
841
842 #[test]
843 fn tampered_hub_checkpoint_fails_verification() {
844 use base64::{engine::general_purpose::URL_SAFE_NO_PAD, Engine};
845 use ed25519_dalek::{Signer, SigningKey};
846
847 let sk = SigningKey::from_bytes(&[1u8; 32]);
848 let pk = sk.verifying_key();
849
850 let mut cp = sample_checkpoint(CheckpointKind::HubOrg);
851 cp.hub_id = "hub://x".into();
852 cp.hub_public_key = URL_SAFE_NO_PAD.encode(pk.to_bytes());
853 cp.signed_at = "2026-04-30T07:00:00Z".into();
854 cp.covered_use_ids = vec!["use_alpha".into()];
855
856 let sig = sk.sign(&cp.canonical_hub_signing_bytes());
857 cp.hub_signature = URL_SAFE_NO_PAD.encode(sig.to_bytes());
858 assert_eq!(verify_hub_checkpoint_signature(&cp), HubCheckpointVerification::Valid);
860
861 cp.covered_use_ids.push("use_smuggled".into());
864 assert_eq!(
865 verify_hub_checkpoint_signature(&cp),
866 HubCheckpointVerification::Tampered,
867 );
868 }
869
870 #[test]
871 fn wrong_key_fails_verification() {
872 use base64::{engine::general_purpose::URL_SAFE_NO_PAD, Engine};
873 use ed25519_dalek::{Signer, SigningKey};
874
875 let sk_real = SigningKey::from_bytes(&[2u8; 32]);
876 let sk_imp = SigningKey::from_bytes(&[3u8; 32]); let pk_imp = sk_imp.verifying_key();
878
879 let mut cp = sample_checkpoint(CheckpointKind::HubOrg);
880 cp.hub_id = "hub://x".into();
881 cp.hub_public_key = URL_SAFE_NO_PAD.encode(pk_imp.to_bytes());
883 cp.signed_at = "2026-04-30T07:00:00Z".into();
884 let sig = sk_real.sign(&cp.canonical_hub_signing_bytes());
885 cp.hub_signature = URL_SAFE_NO_PAD.encode(sig.to_bytes());
886 assert_eq!(
887 verify_hub_checkpoint_signature(&cp),
888 HubCheckpointVerification::Tampered,
889 );
890 }
891
892 #[test]
893 fn malformed_pubkey_or_signature_fails() {
894 let mut cp = sample_checkpoint(CheckpointKind::HubOrg);
895 cp.hub_id = "hub://x".into();
896 cp.hub_public_key = "not-base64!!".into();
897 cp.hub_signature = "also-not-base64".into();
898 cp.signed_at = "2026-04-30T07:00:00Z".into();
899 assert_eq!(
900 verify_hub_checkpoint_signature(&cp),
901 HubCheckpointVerification::Tampered,
902 );
903 }
904}