elicitation 0.10.0

Conversational elicitation of strongly-typed Rust values via MCP
Documentation
//! Integration tests for contract-based elicitation.

use elicitation::{
    ElicitResult,
    contracts::{And, Established, InVariant, Is, both},
};

/// Test that elicit_proven returns a proof
#[tokio::test]
#[cfg_attr(not(feature = "api"), ignore)]
async fn test_string_elicit_proven() -> ElicitResult<()> {
    // This would require actual MCP infrastructure
    // For now, we test the trait signature compiles
    Ok(())
}

/// Test that proofs can be used in downstream functions
#[test]
fn test_proof_composition() {
    #[derive(elicitation::Prop)]
    struct EmailValidated;
    #[derive(elicitation::Prop)]
    struct ConsentObtained;

    // Simulate two proofs
    let email_proof: Established<EmailValidated> = Established::assert();
    let consent_proof: Established<ConsentObtained> = Established::assert();

    // Combine them
    let combined: Established<And<EmailValidated, ConsentObtained>> =
        both(email_proof, consent_proof);

    // Function requiring both proofs
    fn register_user(_email: String, _proof: Established<And<EmailValidated, ConsentObtained>>) {
        // Would register user
    }

    register_user("user@example.com".to_string(), combined);
}

/// Test that Is<T> proofs work with concrete types
#[test]
fn test_inhabitation_proof() {
    fn use_validated_string(_s: String, _proof: Established<Is<String>>) {
        // Would use validated string
    }

    let s = String::from("hello");
    let proof = Established::assert();
    use_validated_string(s, proof);
}

/// Test that proofs are zero-sized
#[test]
fn test_proofs_zero_sized() {
    let string_proof: Established<Is<String>> = Established::assert();
    assert_eq!(std::mem::size_of_val(&string_proof), 0);

    let i32_proof: Established<Is<i32>> = Established::assert();
    assert_eq!(std::mem::size_of_val(&i32_proof), 0);

    #[derive(elicitation::Prop)]
    struct EmailValid;
    let custom_proof: Established<EmailValid> = Established::assert();
    assert_eq!(std::mem::size_of_val(&custom_proof), 0);
}

/// Test InVariant with enum state machine
#[test]
fn test_invariant_state_machine() {
    enum WorkflowState {
        Draft,
        Review,
        Approved,
    }

    struct DraftVariant;
    struct ReviewVariant;
    struct ApprovedVariant;

    // State-specific functions that require variant proofs
    fn edit_draft(
        _state: WorkflowState,
        _proof: Established<InVariant<WorkflowState, DraftVariant>>,
    ) {
        // Can only edit in draft state
    }

    fn submit_for_review(
        _state: WorkflowState,
        _proof: Established<InVariant<WorkflowState, DraftVariant>>,
    ) -> Established<InVariant<WorkflowState, ReviewVariant>> {
        // Transition: Draft → Review
        Established::assert()
    }

    fn approve(
        _state: WorkflowState,
        _proof: Established<InVariant<WorkflowState, ReviewVariant>>,
    ) -> Established<InVariant<WorkflowState, ApprovedVariant>> {
        // Transition: Review → Approved
        Established::assert()
    }

    // Workflow: Draft → Review → Approved
    let draft_proof: Established<InVariant<WorkflowState, DraftVariant>> = Established::assert();
    edit_draft(WorkflowState::Draft, draft_proof);

    let review_proof = submit_for_review(WorkflowState::Draft, draft_proof);
    let approved_proof = approve(WorkflowState::Review, review_proof);

    // Verify we can construct the final state
    let _final_state = (WorkflowState::Approved, approved_proof);

    // Cannot skip states (would fail to compile):
    // approve(WorkflowState::Draft, draft_proof); // Type error!
}