Skip to main content

APAAI Protocol Python SDK — Action → Policy → Evidence

Project description

apaai-client

PyPI version License

Python SDK for APAAI Protocol

Open, vendor-neutral SDK for the APAAI Protocol's Action → Policy → Evidence loop.

  • 📦 Package: apaai-client
  • 🔌 Protocol: HTTP/JSON (/actions, /evidence, /policy)
  • 🧪 Minimal & testable: Class-based API
  • 🧱 License: Apache-2.0

Install

pip install apaai-client

Reference server (for local development):

cd server
npm i && npm run dev    # → http://localhost:8787

Quickstart

from apaai_client import AccountabilityLayer

# Initialize the accountability layer
apaai = AccountabilityLayer(
    endpoint="https://api.apaaiprotocol.org",
    api_key=os.getenv("APAAI_API_KEY")
)

# 1) Propose an action
decision = apaai.propose(
    type="send_email",
    actor={"kind": "agent", "name": "mail-bot"},
    target="mailto:client@acme.com",
    params={"subject": "Proposal"}
)

# 2) Add evidence
apaai.evidence.add(decision["actionId"], [
    {"name": "email_sent", "pass": True, "note": "msgId=123"}
])

with_action Helper

The with_action helper orchestrates the complete flow:

import asyncio
from apaai_client import AccountabilityLayer, with_action

apaai = AccountabilityLayer(
    endpoint="https://api.apaaiprotocol.org",
    api_key=os.getenv("APAAI_API_KEY")
)

async def main():
    await with_action(
        apaai=apaai,
        type="send_email",
        actor={"kind": "agent", "name": "mail-bot"},
        target="mailto:client@acme.com",
        params={"subject": "Proposal"},
        
        on_approval=async (action_data) => {
            # Handle approval workflow
            await apaai.human.approve(action_data["actionId"], "@reviewer")
        },
        
        execute=async () => {
            # Your business logic
            return await send_email({"to": "client@acme.com", "subject": "Proposal"})
        },
        
        evidence_on_success=lambda result: [
            {"name": "email_sent", "pass": True, "note": f"msgId={result['id']}"}
        ],
        evidence_on_error=lambda err: [
            {"name": "email_failed", "pass": False, "note": str(err)}
        ]
    )

asyncio.run(main())

API Reference

AccountabilityLayer Class

apaai = AccountabilityLayer(
    endpoint: Optional[str] = None, 
    api_key: Optional[str] = None
)

Core Methods

  • propose(action) - Propose an action and get a decision
  • evidence(action_id, checks) - Submit evidence for an action
  • policy(action_type?) - Get policy for an action type
  • approve(action_id, approver?) - Approve an action
  • reject(action_id, reason?) - Reject an action
  • get_action(action_id) - Get action details
  • list_actions(filters?) - List actions with filters
  • get_evidence(action_id) - Get evidence for an action
  • set_policy(policy) - Set a policy

Manager Interfaces

  • apaai.policies.evaluate(action_id) - Evaluate policy for an action
  • apaai.policies.enforce(action_type) - Enforce policy for an action type
  • apaai.policies.set(policy) - Set a policy
  • apaai.human.approve(action_id, approver?) - Approve an action
  • apaai.human.reject(action_id, reason?) - Reject an action
  • apaai.evidence.add(action_id, checks) - Add evidence for an action
  • apaai.evidence.get(action_id) - Get evidence for an action
  • apaai.actions.get(action_id) - Get action details
  • apaai.actions.list(filters?) - List actions with filters

Examples

Basic Flow

from apaai_client import AccountabilityLayer

apaai = AccountabilityLayer(endpoint="http://localhost:8787")

# Propose action
decision = apaai.propose(
    type="send_email",
    actor={"kind": "agent", "name": "mail-bot"},
    target="mailto:client@acme.com",
    params={"subject": "Proposal"}
)

# Handle approval if required
if decision["status"] == "requires_approval":
    apaai.approve(decision["actionId"], "@reviewer")

# Submit evidence
apaai.evidence.add(decision["actionId"], [
    {"name": "email_sent", "pass": True, "note": "msgId=123"}
])

Using Manager Interfaces

# Policy management
policy = apaai.policies.enforce("send_email")
apaai.policies.set({"rules": [...]})

# Human-in-the-loop
apaai.human.approve(action_id, "@reviewer")
apaai.human.reject(action_id, "Invalid recipient")

# Evidence management
apaai.evidence.add(action_id, [
    {"name": "email_sent", "pass": True, "note": "msgId=123"}
])
evidence = apaai.evidence.get(action_id)

# Action management
action = apaai.actions.get(action_id)
actions = apaai.actions.list({"type": "send_email"})

Types

from typing import Dict, List, Optional, Any

Actor = Dict[str, Any]  # {"kind": "agent", "name": "mail-bot", "provider": "openai"}

Action = Dict[str, Any]  # {
    "id": str,
    "timestamp": str,
    "type": str,
    "actor": Actor,
    "target": Optional[str],
    "params": Optional[Dict[str, Any]],
    "status": Optional[str],
    "checks": Optional[List[str]]
}

Check = Dict[str, Any]  # {"name": str, "pass": bool, "note": Optional[str]}

Evidence = Dict[str, Any]  # {"actionId": str, "checks": List[Check]}

Decision = Dict[str, Any]  # {"status": str, "checks": Optional[List[str]]}

PolicyRule = Dict[str, Any]  # {
    "when": Optional[Dict[str, str]],  # {"action": str} or {"actionType": str}
    "require": Optional[List[str]],
    "mode": Optional[str]  # "enforce" | "observe"
}

Policy = Dict[str, Any]  # {"rules": List[PolicyRule]}

Testing

# Run tests
python -m pytest tests/

# Run tests with coverage
python -m pytest tests/ --cov=apaai_client

# Run specific test file
python -m pytest tests/test_client.py

Build & Publish

# Build the package
python -m build

# Publish to PyPI
python -m twine upload dist/*

License

Apache-2.0

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

apaai-0.1.1.tar.gz (10.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

apaai-0.1.1-py3-none-any.whl (9.8 kB view details)

Uploaded Python 3

File details

Details for the file apaai-0.1.1.tar.gz.

File metadata

  • Download URL: apaai-0.1.1.tar.gz
  • Upload date:
  • Size: 10.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.13.7

File hashes

Hashes for apaai-0.1.1.tar.gz
Algorithm Hash digest
SHA256 a52b909d13b5920488a1d966c80df7d6add8c7d444450e3a017b8f44f1bd9b19
MD5 37626295e8fd55941decfc2440f188d3
BLAKE2b-256 1f636dc51d711e15f9c85af7a4e3a7b5c0f02f55b0bc906213a4f96602de19e4

See more details on using hashes here.

File details

Details for the file apaai-0.1.1-py3-none-any.whl.

File metadata

  • Download URL: apaai-0.1.1-py3-none-any.whl
  • Upload date:
  • Size: 9.8 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.13.7

File hashes

Hashes for apaai-0.1.1-py3-none-any.whl
Algorithm Hash digest
SHA256 0cb1ec4457c4935410b48dc66cd81c5e0dafd8b7a3c4114b1c3235a923787b81
MD5 c9b6775614651d7f8c52a555fc8df575
BLAKE2b-256 6ae391b9a2d51bd79249a1f90cb05fe1e0ce2aa59ab18bea306c5aa2e72d4e23

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page