delegated 0.1.1

Fail-closed trust evaluation for agentic AI systems — delegation tokens, policy enforcement, and audit for agent-to-agent and human-to-agent workflows.
Documentation
use crate::adapters::a2a::{A2aProtocolRequest, A2aProtocolResponse};
use crate::adapters::guard::{AdapterGuardConfig, enter_adapter_guard};
use crate::audit::AuditSink;
use crate::engine_async::evaluate_and_audit_with_async_state;
use crate::models::{HostContext, RequestEnvelope};
use crate::revocation_async::AsyncTrustStateStore;
use crate::wire::SHARED_CLAIMS_KIND;
use chrono::{DateTime, Utc};
use serde_json::json;

pub async fn handle_a2a_request_with_async_state(
    raw_body: &str,
    now: DateTime<Utc>,
    sink: &dyn AuditSink,
    trust_state: &dyn AsyncTrustStateStore,
    host_context: &HostContext,
) -> A2aProtocolResponse {
    handle_a2a_request_with_async_state_and_guard_config(
        raw_body,
        now,
        sink,
        trust_state,
        &AdapterGuardConfig::default(),
        host_context,
    )
    .await
}

pub async fn handle_a2a_request_with_async_state_and_guard_config(
    raw_body: &str,
    now: DateTime<Utc>,
    sink: &dyn AuditSink,
    trust_state: &dyn AsyncTrustStateStore,
    guard_config: &AdapterGuardConfig,
    host_context: &HostContext,
) -> A2aProtocolResponse {
    let request: A2aProtocolRequest = match serde_json::from_str(raw_body) {
        Ok(value) => value,
        Err(error) => {
            return A2aProtocolResponse {
                message_id: "unknown".to_string(),
                status: "error".to_string(),
                result: None,
                error: Some(
                    json!({"stage":"a2a_adapter","reason":format!("malformed A2A request: {error}")}),
                ),
            };
        }
    };

    if request.trust_claims.kind != SHARED_CLAIMS_KIND {
        return A2aProtocolResponse {
            message_id: request.message_id,
            status: "error".to_string(),
            result: None,
            error: Some(
                json!({"stage":"a2a_adapter","reason":format!("trust_claims.kind must equal {SHARED_CLAIMS_KIND}")}),
            ),
        };
    }

    let _guard_lease = match enter_adapter_guard(
        &request.trust_claims.agent_id,
        &request.trust_claims.delegator_id,
        now,
        guard_config,
    ) {
        Ok(lease) => lease,
        Err(violation) => {
            return A2aProtocolResponse {
                message_id: request.message_id,
                status: "denied".to_string(),
                result: None,
                error: Some(json!({
                    "stage": "adapter_guard",
                    "reason": violation.reason
                })),
            };
        }
    };

    let envelope: RequestEnvelope = request.trust_claims.into();
    let raw_envelope = match serde_json::to_value(envelope) {
        Ok(value) => value,
        Err(error) => {
            return A2aProtocolResponse {
                message_id: request.message_id,
                status: "error".to_string(),
                result: None,
                error: Some(
                    json!({"stage":"a2a_adapter","reason":format!("failed to encode request envelope: {error}")}),
                ),
            };
        }
    };

    match evaluate_and_audit_with_async_state(&raw_envelope, now, sink, trust_state, host_context)
        .await
    {
        Ok(decision) if decision.allowed => A2aProtocolResponse {
            message_id: request.message_id,
            status: "ok".to_string(),
            result: Some(json!({
                "allowed": true,
                "stage": decision.stage,
                "reason": decision.reason
            })),
            error: None,
        },
        Ok(decision) => A2aProtocolResponse {
            message_id: request.message_id,
            status: "denied".to_string(),
            result: None,
            error: Some(json!({
                "stage": decision.stage,
                "reason": decision.reason
            })),
        },
        Err(error) => A2aProtocolResponse {
            message_id: request.message_id,
            status: "error".to_string(),
            result: None,
            error: Some(
                json!({"stage":"audit_sink","reason":format!("failed to write audit event: {error}")}),
            ),
        },
    }
}

#[cfg(test)]
mod tests {
    use super::*;
    use crate::audit::JsonlFileAuditSink;
    use crate::crypto::{
        TOKEN_SIGNATURE_ALG_ED25519, sign_delegation_token, sign_identity_document,
    };
    use crate::models::{
        AgentEndpoint, AgentIdentityDocument, DelegationToken, PublicKeyRecord, RequestEnvelope,
        RuntimeContext, TrustProfile,
    };
    use crate::revocation_async::InMemoryAsyncTrustState;
    use crate::wire::SharedTrustClaims;
    use base64ct::{Base64UrlUnpadded, Encoding};
    use chrono::TimeZone;
    use ed25519_dalek::SigningKey;
    use serde_json::json;
    use std::sync::atomic::{AtomicU64, Ordering};

    static REQUEST_COUNTER: AtomicU64 = AtomicU64::new(1);

    fn now() -> DateTime<Utc> {
        Utc.with_ymd_and_hms(2026, 6, 1, 20, 20, 0)
            .single()
            .expect("valid test timestamp")
    }

    fn unique_id() -> String {
        let counter = REQUEST_COUNTER.fetch_add(1, Ordering::Relaxed);
        let nanos = std::time::SystemTime::now()
            .duration_since(std::time::UNIX_EPOCH)
            .expect("time should be after epoch")
            .as_nanos();
        format!("{counter}_{nanos}")
    }

    fn claims(nonce: &str) -> SharedTrustClaims {
        let unique_id = unique_id();
        let key = SigningKey::from_bytes(&[77u8; 32]);
        let mut identity = AgentIdentityDocument {
            spec_version: "0.1".to_string(),
            kind: "AgentIdentityDocument".to_string(),
            agent_id: "agent:example:scheduler:v1".to_string(),
            display_name: None,
            owner_id: "org:example".to_string(),
            issuer: "https://trust.example.ai".to_string(),
            identity_type: "spiffe".to_string(),
            subject: "spiffe://example.ai/agents/scheduler".to_string(),
            public_keys: vec![PublicKeyRecord {
                kid: "key-2026-01".to_string(),
                kty: "OKP".to_string(),
                crv: Some(TOKEN_SIGNATURE_ALG_ED25519.to_string()),
                x: Some(Base64UrlUnpadded::encode_string(
                    &key.verifying_key().to_bytes(),
                )),
            }],
            supported_protocols: vec!["a2a".to_string()],
            supported_auth_methods: vec!["delegation_token".to_string()],
            capabilities: None,
            endpoints: vec![AgentEndpoint {
                protocol: "a2a".to_string(),
                url: "https://agents.example.ai/scheduler/a2a".to_string(),
            }],
            attestation: None,
            created_at: Utc
                .with_ymd_and_hms(2026, 6, 1, 20, 0, 0)
                .single()
                .expect("valid timestamp"),
            expires_at: Utc
                .with_ymd_and_hms(2026, 6, 8, 20, 0, 0)
                .single()
                .expect("valid timestamp"),
            signature: String::new(),
        };
        identity.signature = sign_identity_document(&identity, &key).expect("identity signing");

        let mut token = DelegationToken {
            spec_version: "0.1".to_string(),
            kind: "DelegationToken".to_string(),
            token_id: format!("dlg_a2a_async_{unique_id}"),
            issuer: "https://trust.example.ai".to_string(),
            agent_id: "agent:example:scheduler:v1".to_string(),
            delegator_id: "user:jake-abendroth".to_string(),
            owner_id: "org:example".to_string(),
            audience: vec!["tool:google-calendar".to_string()],
            allowed_actions: vec!["calendar.create_event".to_string()],
            resource_constraints: None,
            max_spend: None,
            max_delegation_depth: Some(0),
            issued_at: Utc
                .with_ymd_and_hms(2026, 6, 1, 20, 10, 0)
                .single()
                .expect("valid timestamp"),
            expires_at: Utc
                .with_ymd_and_hms(2026, 6, 1, 20, 40, 0)
                .single()
                .expect("valid timestamp"),
            intent: None,
            nonce: nonce.to_string(),
            key_id: "key-2026-01".to_string(),
            signature_alg: TOKEN_SIGNATURE_ALG_ED25519.to_string(),
            signature: String::new(),
        };
        token.signature = sign_delegation_token(&token, &key).expect("token signing");
        let envelope = RequestEnvelope {
            spec_version: "0.1".to_string(),
            kind: "TrustRequestEnvelope".to_string(),
            request_id: Some(format!("req_a2a_async_{unique_id}")),
            profile: TrustProfile::Developer,
            agent_id: "agent:example:scheduler:v1".to_string(),
            delegator_id: "user:jake-abendroth".to_string(),
            audience: "tool:google-calendar".to_string(),
            action: "calendar.create_event".to_string(),
            resource: None,
            runtime_context: RuntimeContext::default(),
            identity_document: Some(identity),
            token,
        };
        envelope.into()
    }

    #[tokio::test]
    async fn async_a2a_allows_valid_message() {
        let nonce = format!(
            "nonce-a2a-async-{}",
            std::time::SystemTime::now()
                .duration_since(std::time::UNIX_EPOCH)
                .expect("epoch")
                .as_nanos()
        );
        let req = A2aProtocolRequest {
            message_id: "msg-a2a-async-1".to_string(),
            protocol_version: "2026-06-01".to_string(),
            message_type: "task.request".to_string(),
            trust_claims: claims(&nonce),
            payload: json!({"task": "schedule"}),
        };
        let state = InMemoryAsyncTrustState::new();
        let path = std::env::temp_dir().join(format!(
            "delegated_a2a_async_{}.jsonl",
            std::time::SystemTime::now()
                .duration_since(std::time::UNIX_EPOCH)
                .expect("epoch")
                .as_nanos()
        ));
        let sink = JsonlFileAuditSink::new(path.clone());
        let response = handle_a2a_request_with_async_state(
            &serde_json::to_string(&req).expect("serialization should work"),
            now(),
            &sink,
            &state,
            &HostContext::default(),
        )
        .await;
        assert_eq!(response.status, "ok");
        std::fs::remove_file(path).expect("temporary audit file should be removable");
    }

    #[tokio::test]
    async fn async_a2a_blocks_nonce_replay() {
        let nonce = format!(
            "nonce-a2a-async-replay-{}",
            std::time::SystemTime::now()
                .duration_since(std::time::UNIX_EPOCH)
                .expect("epoch")
                .as_nanos()
        );
        let req = A2aProtocolRequest {
            message_id: "msg-a2a-async-replay".to_string(),
            protocol_version: "2026-06-01".to_string(),
            message_type: "task.request".to_string(),
            trust_claims: claims(&nonce),
            payload: json!({}),
        };
        let state = InMemoryAsyncTrustState::new();
        let path = std::env::temp_dir().join(format!(
            "delegated_a2a_async_replay_{}.jsonl",
            std::time::SystemTime::now()
                .duration_since(std::time::UNIX_EPOCH)
                .expect("epoch")
                .as_nanos()
        ));
        let sink = JsonlFileAuditSink::new(path.clone());
        let serialized = serde_json::to_string(&req).expect("serialization should work");
        let first = handle_a2a_request_with_async_state(
            &serialized,
            now(),
            &sink,
            &state,
            &HostContext::default(),
        )
        .await;
        let second = handle_a2a_request_with_async_state(
            &serialized,
            now(),
            &sink,
            &state,
            &HostContext::default(),
        )
        .await;
        assert_eq!(first.status, "ok");
        assert_eq!(second.status, "denied");
        std::fs::remove_file(path).expect("temporary audit file should be removable");
    }
}