Official Humanos API SDK for Python with automatic request signing and webhook verification
Project description
Humanos SDK for Python
Official Python SDK for the Humanos API. Auto-signs every request, verifies and decrypts webhooks, and ships full type definitions (pydantic models).
Concepts
Humanos gives your agent a cryptographic permission slip — issued by the user, scoped to a specific action, and verified at runtime.
- Action — a reusable policy template you publish through the dashboard. Declares
userParams(values the human approves),executionParams(what the agent must supply at execution time), and the rules that compare them. - Mandate — a signed W3C credential (type
POLICY) the user issues to your agent. References one action plus theuserParamsvalues that bind this specific approval. - Verifiable Presentation (VP) — a short-lived, signed proof derived from a mandate. The agent presents one each time it wants to act.
- Verify — Humanos checks the VP's signatures, expiry, revocation status, and the action's rules against the agent's
executionParams. Returns allow or deny. - Revoke — kills a mandate. Any further VP issuance or
verifyis denied.
Lifecycle
A mandate's lifetime breaks into two flows: request and approval (one-time, gets you a mandate ID) and issuance and verification (repeats every time the agent acts).
Flow 1 — Request and approval
┌──────────┐ ┌──────────────┐ ┌──────────┐ ┌──────────┐ ┌─ accepts ─▶ mandate issued
│ Action │ ─▶ │ Agent needs │ ─▶ │ Agent │ ─▶ │ User │ ─▶ ─────┤
│ defined │ │ permission │ │ requests │ │ decides │ └─ rejects ─▶ flow ends
└──────────┘ └──────────────┘ └──────────┘ └──────────┘
step 1 step 2 step 3
Flow 2 — Issuance and verification
┌──────────┐ ┌──────────────────────────────┐ ┌─ true ─▶ allow (agent acts)
│ Agent │ ─▶ │ Humanos verifies VP against │ ─▶ ─────┤
│issues VP │ │ executionParams │ └─ false ─▶ deny (blocked)
└──────────┘ └──────────────────────────────┘
step 4 step 5
Installation
pip install humanos
Requires Python 3.8 or newer.
Configuration
Sign up at humanos.id, then grab two sets of credentials from the dashboard:
- API credentials — Settings → API Keys. Copy the API Key and Signature Secret.
- Webhook credentials — Settings → Webhooks. Copy the Webhook Signature Secret, Webhook Encryption Secret, and Webhook Encryption Salt.
Add them to your .env:
HUMANOS_API_KEY=<your-api-key>
HUMANOS_SIGNATURE_SECRET=<your-signature-secret>
HUMANOS_WEBHOOK_SIGNATURE_SECRET=<your-webhook-signature-secret>
HUMANOS_WEBHOOK_ENCRYPTION_SECRET=<your-webhook-encryption-secret>
HUMANOS_WEBHOOK_ENCRYPTION_SALT=<your-webhook-encryption-salt>
Everything else (action IDs, mandate IDs, etc.) is application data — keep it in code or your database, not in .env.
Quick start
import os
from humanos_sdk import HumanosClient, HumanosClientConfig
client = HumanosClient(HumanosClientConfig(
base_path="https://api.humanos.id",
api_key=os.environ["HUMANOS_API_KEY"],
signature_secret=os.environ["HUMANOS_SIGNATURE_SECRET"],
))
# Sanity check — confirms the API key, signature secret, and signing handshake.
requests = client.requests.list()
print(requests)
A 200 OK with a (possibly empty) list means you're wired up. The walkthrough below exercises the rest of the flow.
Integration walkthrough
Six steps, matching the lifecycle diagram above. Run through them once with a test action to get an end-to-end feel.
1. Define an action
In the dashboard, create an action. An action has three parts:
executionParams— the values the agent supplies atverify()time.userParams— the constants the user pins at decision time.- Rules — deterministic CEL expressions comparing the two above.
Example:
| Part | Field | Type |
|---|---|---|
executionParams |
amount |
number |
category |
string | |
userParams |
maxAmount |
number |
allowedCategories |
array<string> |
Rules then express things like executionParams.amount <= userParams.maxAmount and executionParams.category in userParams.allowedCategories.
Once defined, publish the action and copy its ID. Hold the ID as a constant in your code or store it alongside the rule it represents.
2. Issue a mandate request
Ask Humanos to issue a mandate to a user. The request bundles the contact, the action ID, and the userParams values the user is being asked to approve.
from humanos_sdk_generated.models import (
GenerateRequestDto,
CredentialDto,
ActionRefDto,
)
action_id = "urn:via:action:<uuid>" # from step 1
request = client.requests.create(
GenerateRequestDto(
contacts=["user@example.com"],
security_level="CONTACT",
credentials=[
CredentialDto(
scope="agent.execute",
type="POLICY",
name="Action mandate", # required, shown to the user
action=ActionRefDto(
action_urn=action_id,
user_params={
"maxAmount": 10000,
"allowedCategories": ["BOOKS", "OFFICE"],
},
),
)
],
)
)
print("Request ID:", request.id)
Persist request.id if you want to track pending approvals. The mandate ID itself becomes available once the user approves (next step).
3. User approval and mandate issuance
Humanos handles the user-facing approval flow. The user reviews the userParams from step 2 and either approves or rejects.
Identity verification. The security code (OTP) is always delivered by email or SMS — that's how Humanos proves the user is reachable at the contact you supplied.
Where the approval UI shows up. Two options:
- Hosted (default) — the email or SMS message contains a link to the Humanos-hosted approval page. The user clicks through, reviews, decides.
- Embedded iframe — your application embeds the Humanos approval UI directly. The user never leaves your app. See the iframe integration guide for setup.
If KYC is required on the action, the user completes that first. If the user rejects (or KYC fails), no mandate is issued.
On accept, Humanos issues the mandate and emits a credential webhook event. The mandate ID — urn:via:credential:… — is the durable artifact: persist it on the rule record in your database (e.g., rule_id → mandate_id).
The SDK ships Flask and FastAPI handler factories that verify signatures and decrypt payloads automatically. FastAPI:
import os
from fastapi import FastAPI, Request
from humanos_sdk import WebhookConfig, create_fastapi_webhook_handler
app = FastAPI()
config = WebhookConfig(
signature_secret=os.environ["HUMANOS_WEBHOOK_SIGNATURE_SECRET"],
encryption_secret=os.environ["HUMANOS_WEBHOOK_ENCRYPTION_SECRET"],
encryption_salt=os.environ["HUMANOS_WEBHOOK_ENCRYPTION_SALT"],
)
def on_event(payload):
event_type = payload["eventType"]
if event_type == "credential":
if payload["decision"]["action"] != "accept":
# user rejected — mark rule unenforceable, notify operator
return
mandate_id = payload["credential"]["id"]
# persist: rule_id → mandate_id
elif event_type == "identity":
# KYC / identity verification completed
pass
elif event_type == "otp.failed":
# user failed OTP — possible fraud signal
pass
elif event_type == "test":
# "Send test event" from the dashboard
pass
handler = create_fastapi_webhook_handler(config, on_event)
@app.post("/webhook")
async def webhook(request: Request):
return await handler(request)
Flask is symmetric — use create_flask_webhook_handler(config, on_event) and call the returned function from your route.
The credential payload shape:
{
"eventType": "credential",
"requestId": str,
"internalId": str | None,
"issuerDid": str,
"user": {"contact": str, "id": str, "internalId": str | None},
"credential": CredentialEntity, # full W3C VC; ["id"] is the mandate ID
"decision": {"action": "accept" | "reject", "date": str}, # ISO 8601
}
For local development, expose your server with ngrok:
ngrok http 8000
Set the ngrok URL (e.g. https://xxxx.ngrok-free.app/webhook) under Settings → Webhooks.
If you're using the iframe channel, the same credential payload is also delivered via window.postMessage to the parent window — useful for live UI updates without a backend round-trip.
Dev shortcut: during development you can copy the mandate ID directly from the dashboard's activity table (look for the
MANDATE_ISSUEentry) instead of wiring a webhook.
4. Agent issues a VP
When the agent wants to act, your backend (or whichever component enforces the rule) asks Humanos for a fresh Verifiable Presentation bound to the mandate ID captured in step 3. VPs are short-lived and single-purpose — issue a new one for every verify.
from humanos_sdk_generated.models import CreatePresentationDto
mandate_id = "urn:via:credential:<id>" # captured in step 3
vp = client.credentials.issue_vp(
mandate_id,
CreatePresentationDto(
# target_verifier="did:web:your-verifier.example", # optional, see below
),
)
Field-by-field:
mandate_id(first positional arg) — the mandate ID captured in step 3.target_verifier(optional body field) — DID of the intended verifier. When provided, the VP is bound to that audience viaproof.domainplus a challenge nonce.
The response is a PresentationResponseEntity — presentation and receipt. Pass vp.presentation to step 5.
5. Humanos verifies the VP
Hand the VP plus the agent's executionParams to verify(). Humanos checks four things: signatures, expiry, revocation status, and rule compliance. 200 OK means allow; 403 means at least one check failed (the response body names the failing rule under evaluations).
from humanos_sdk_generated.models import VerifyPresentationDto
client.credentials.verify(
VerifyPresentationDto(
presentation=vp.presentation,
execution_params={"amount": 5000, "category": "BOOKS"},
)
)
Field-by-field:
presentation— the signed VP from step 4. Read it fromvp.presentation.execution_params— what the agent wants to do. Field names must match those declared on the action and are referenced inside rules asexecutionParams.<field>.
Expected outcomes for the example action:
| Case | execution_params |
Result |
|---|---|---|
| In-bounds | {"amount": 5000, "category": "BOOKS"} |
allow (200) + signed receipt |
| Out-of-bounds | {"amount": 50000, "category": "FLIGHTS"} |
deny (403) — body names the failing rule(s) |
6. Revoke a mandate
Mandates are immutable — to retire one (rule deletion, rule update, user pulling consent), call credentials.revoke(). Once revoked, the mandate is dead: issue_vp errors with credential_revoked, and any VP still held by an agent fails verify() with the same reason. Stale rules cannot be enforced; that's the safety property.
from humanos_sdk_generated.models import RevokeCredentialDto
client.credentials.revoke(
mandate_id,
revoke_credential_dto=RevokeCredentialDto(reason="user_initiated"),
)
Field-by-field:
credential_id(first arg) — the mandate ID.reason— free-text label recorded on the credential and in theMANDATE_REVOKEDreceipt. Recommended values from the VIA protocol:user_initiated,organization_policy,system_expiry.
The response is a RevokeCredentialEntity — credential_id, status="REVOKED", revoked_at, and a MANDATE_REVOKED receipt. Store it for audit if useful. Enforcement uses the revocation status itself, not the receipt.
Humanos audit trail
For compliance, Humanos persists every consequential event in a mandate's lifecycle as an immutable, cryptographically signed record. You don't need to log these yourself — they're available for query at any time via client.activity.list().
| Event | Trigger | Captured |
|---|---|---|
| Mandate issued | User accepts a request | Action ID, userParams, signed credential, timestamp |
| Mandate revoked | credentials.revoke() succeeds |
Mandate ID, reason, timestamp |
| Mandate canceled | Request canceled before approval | Request ID, canceler, timestamp |
| Human decision: accept | User approves a request | User, request, OTP channel (email or SMS), UI surface (hosted / iframe) |
| Human decision: reject | User rejects a request | User, request, OTP channel, UI surface |
| VP issue | credentials.issue_vp() succeeds |
Mandate ID, VP, target verifier, receipt |
| VP issue denied | credentials.issue_vp() rejected |
Mandate ID, reason code (e.g., credential_revoked) |
| Verify accept | credentials.verify() returns 200 |
VP, executionParams, rule evaluations, signed receipt |
| Verify deny | credentials.verify() returns 403 |
VP, executionParams, failing rule(s), signed receipt |
Webhook details
The Flask / FastAPI handler factories decrypt and verify the payload, then call your callback with the decoded dict. Switch on payload["eventType"] to dispatch:
| Event | Trigger |
|---|---|
credential |
A credential request was approved or rejected. credential["id"] is the mandate ID. |
identity |
An identity / KYC check completed. |
otp.failed |
A user failed OTP entry. |
test |
"Send test event" from the dashboard. |
Operational notes
- Send a test event from Settings → Webhooks before going live — confirms your URL is reachable and your handler decodes the payload.
- Retries: failed deliveries are retried with backoff. Make your handler idempotent (key off
requestId+eventType). - Order: events for unrelated requests may arrive out of order. Within a single request,
credentialprecedes any follow-up events.
API reference
Every method below is fully typed via pydantic; signatures live in the generated package.
# Requests — credential request lifecycle
client.requests.list() # paginate / filter your requests
client.requests.create(generate_request_dto) # issue a credential request to a user
client.requests.detail(request_id) # full request including credentials and subjects
client.requests.cancel(request_id) # cancel a pending request
client.requests.resend_otp(request_id) # resend OTP via the original channel
# Credentials — mandates and verification
client.credentials.detail(credential_id) # fetch a credential by ID
client.credentials.evidence(evidence_id) # download attached evidence
client.credentials.issue_vp(vc_id, create_presentation_dto) # issue a fresh VP from a mandate
client.credentials.verify(verify_presentation_dto) # evaluate a VP at runtime
client.credentials.revoke(credential_id, revoke_credential_dto=...) # permanently revoke a mandate
# Actions — published policy templates
client.actions.list()
client.actions.versions(action_id)
# Activity — audit log
client.activity.list()
# Approval workflows (forms, consents, documents)
client.approval.list()
client.approval.workflows()
# Users
client.users.create(create_subject_dto_list)
client.users.detail(contact=...)
# DID resolution
client.did.resolve(did)
API versioning
Humanos uses date-based API versions (e.g., 2026-03-24). The SDK pins each release to a default version and sends it as an api-version header on every request — your integration stays isolated from API changes, and you upgrade by bumping the SDK on your own schedule.
The current default is exported as API_VERSION:
from humanos_sdk import API_VERSION
print(API_VERSION) # "2026-03-24"
Every method also accepts an optional api_version keyword argument for per-call overrides — useful when testing a new version against a single endpoint before bumping globally:
client.credentials.verify(
VerifyPresentationDto(
presentation=vp.presentation,
execution_params={"amount": 5000, "category": "BOOKS"},
),
api_version="2025-12-01", # api version for this call only
)
To pin globally, subclass SignedApiClient and override the api-version header — or upgrade the SDK release once you're ready to migrate.
Action versions are independent of API versions. When you republish an action in the dashboard, mandates already issued against the older version keep referencing that version — they don't break. List published versions with client.actions.versions(action_id).
Error handling
The SDK raises ApiException (from humanos_sdk_generated.exceptions) on non-2xx responses. The exception carries the HTTP status and response body:
from humanos_sdk_generated.exceptions import ApiException
try:
client.requests.create(generate_request_dto)
except ApiException as e:
print("Status:", e.status)
print("Body:", e.body)
Troubleshooting
401 on every request. Signature mismatch. Double-check the signature secret matches the dashboard exactly — no trailing whitespace or stray newline.
400 on requests.create() with "name is required". Each CredentialDto needs a name field. The example in step 2 uses "Action mandate".
Webhook never fires. Confirm the URL in Settings → Webhooks matches your live endpoint. Send a test event from the dashboard. For local dev, the ngrok tunnel must be open when the user approves.
Webhook fires but the handler errors with a signature mismatch. Make sure your framework hands the raw request body to process_webhook (not a parsed/reformatted version). The provided Flask / FastAPI handlers do this correctly.
verify() returns 403 with no obvious failing rule. Check that the execution_params field names exactly match the action's declared fields. A typo silently fails rule evaluation.
issue_vp() returns 400 after revoke. Expected — once a mandate is revoked, VP issuance is blocked at source with credential_revoked. Existing VPs also fail verify() for the same reason.
Documentation & related
- Full API documentation
- Humanos website
- Sister SDKs: TypeScript, C#
Support
- Email: tech@humanos.tech
License
MIT License - see LICENSE file for details.
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
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file humanos-1.0.4.tar.gz.
File metadata
- Download URL: humanos-1.0.4.tar.gz
- Upload date:
- Size: 88.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5114298086735f7e8c9ee2bfcafd677baaca48edc6a6c15bcbd24e17a50d9d68
|
|
| MD5 |
a913284a2a9dde400ea7200d0e0ace51
|
|
| BLAKE2b-256 |
84e54e619d14952322bfb84029b5201062e50fd1b8e9b007f55998d4dba69d28
|
Provenance
The following attestation bundles were made for humanos-1.0.4.tar.gz:
Publisher:
publish-python.yml on Humanos-App/humanos-sdks
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
humanos-1.0.4.tar.gz -
Subject digest:
5114298086735f7e8c9ee2bfcafd677baaca48edc6a6c15bcbd24e17a50d9d68 - Sigstore transparency entry: 1586357988
- Sigstore integration time:
-
Permalink:
Humanos-App/humanos-sdks@d569e576be245c02f6a57a44a5aae53721f825ba -
Branch / Tag:
refs/tags/v1.0.4 - Owner: https://github.com/Humanos-App
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish-python.yml@d569e576be245c02f6a57a44a5aae53721f825ba -
Trigger Event:
release
-
Statement type:
File details
Details for the file humanos-1.0.4-py3-none-any.whl.
File metadata
- Download URL: humanos-1.0.4-py3-none-any.whl
- Upload date:
- Size: 233.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c3bb40c11ef45e98315c859016176c733d911c53593ec3f8f97e5fb4eeca50e9
|
|
| MD5 |
d89c49b505f56e6556b03e9ed5b992ed
|
|
| BLAKE2b-256 |
5807cfc1f6f3015b144f5b3411c2284a349c40adcf18aa424f88efc246c0f0ff
|
Provenance
The following attestation bundles were made for humanos-1.0.4-py3-none-any.whl:
Publisher:
publish-python.yml on Humanos-App/humanos-sdks
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
humanos-1.0.4-py3-none-any.whl -
Subject digest:
c3bb40c11ef45e98315c859016176c733d911c53593ec3f8f97e5fb4eeca50e9 - Sigstore transparency entry: 1586358032
- Sigstore integration time:
-
Permalink:
Humanos-App/humanos-sdks@d569e576be245c02f6a57a44a5aae53721f825ba -
Branch / Tag:
refs/tags/v1.0.4 - Owner: https://github.com/Humanos-App
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish-python.yml@d569e576be245c02f6a57a44a5aae53721f825ba -
Trigger Event:
release
-
Statement type: