Castle protects your users from account compromise
Project description
The official Python SDK for Castle. Castle analyzes user behavior in web and mobile apps to stop fraud before it happens.
This package is a thin wrapper around the Castle HTTP API. It exposes risk assessment, event logging, Lists, Privacy (GDPR), Events (enterprise), and webhook verification. See the API reference for supported events and payload shapes.
Requirements
Python 3.9 or newer
A Castle API secret
Installation
pip install castle
Quick start
import os
from castle.configuration import configuration
from castle.client import Client
configuration.api_secret = os.environ['CASTLE_API_SECRET']
client = Client.from_request(request)
verdict = client.risk({
'event': '$login',
'status': '$succeeded',
'request_token': request.POST.get('castle_request_token'),
'user': {'id': '12345', 'email': 'user@example.com'},
})
action = verdict.get('policy', {}).get('action') or verdict.get('action')
if action == 'deny':
# block the user
pass
elif action == 'challenge':
# send 2FA / additional verification
pass
else:
# allow
pass
Client.from_request builds request context (IP, headers, client id) from a framework request object. See Advanced configuration for header allow/deny lists and proxy chains.
Configuration
The minimal, recommended setup:
import os
from castle.configuration import configuration
configuration.api_secret = os.environ['CASTLE_API_SECRET']
# Behavior when Castle's API is unreachable or returns a 5xx.
# One of: allow (default), deny, challenge, throw
configuration.failover_strategy = 'allow'
# Request timeout in milliseconds (default: 1000).
# RequestError is raised on timeout.
configuration.request_timeout = 1000
Logging
import logging
from castle.configuration import configuration
configuration.logger = logging.getLogger('castle')
The logger only needs to respond to info. Each request and response is logged with sensitive values stripped.
Multi-environment / multi-tenant
Most apps only need the global configuration singleton, but you can also create standalone Configuration instances and pass them per call via APIRequest:
from castle.configuration import Configuration
from castle.api_request import APIRequest
from castle.commands.risk import CommandsRisk
config = Configuration()
config.api_secret = os.environ['CASTLE_API_SECRET_TENANT_A']
APIRequest(config).call(CommandsRisk(context).call({
'event': '$login',
'status': '$succeeded',
'request_token': '<token>',
'user': {'id': '1234'},
}))
Usage
See Castle documentation and the API reference for endpoint details, event types, and integration guides.
Advanced configuration
The defaults work for most deployments. The options below only matter if you have a non-trivial proxy chain or strict header policies.
Header allow/deny lists
By default the SDK sends every HTTP header except Cookie and Authorization. Castle uses these headers to fingerprint the request.
from castle.configuration import configuration, DEFAULT_ALLOWLIST
# Always-blocked headers (in addition to Cookie/Authorization).
configuration.denylisted = ['HTTP-X-Internal-Header']
# Strict allow-list mode. Headers outside the list are scrubbed,
# except User-Agent which is always preserved.
configuration.allowlisted = DEFAULT_ALLOWLIST
Header names are case-insensitive and accept both _ and - as separators. A leading HTTP_ prefix is stripped automatically.
Client IP detection
Castle needs the original client IP, not the IP of your proxy or load balancer. The SDK reads X-Forwarded-For and Remote-Addr by default; pick one of the strategies below:
from castle.configuration import configuration, TRUSTED_PROXIES
# 1. Custom header (e.g. Cloudflare's Cf-Connecting-Ip).
configuration.ip_headers = ['Cf-Connecting-Ip']
# 2. Static, known proxy IPs (strings or regexes).
configuration.trusted_proxies = ['10.0.0.1']
# 3. Ephemeral proxies but known chain depth.
configuration.trusted_proxy_depth = 2
# 4. Last resort: trust the entire X-Forwarded-For chain.
# Warning: vulnerable to header spoofing if a malicious proxy is in path.
configuration.trust_proxy_chain = False
Use either trusted_proxies or trusted_proxy_depth, not both. Private/loopback ranges in TRUSTED_PROXIES are always considered trusted.
Optional settings
from castle.configuration import configuration
# Override the API base URL (default: https://api.castle.io/v1)
# configuration.base_url = 'https://api.castle.io/v1'
Signature
Secure mode signs user identifiers on the server:
from castle.secure_mode import signature
signature(user_id)
Exceptions
All exceptions inherit from CastleError. The most useful ones:
ConfigurationError — the SDK is misconfigured (missing API secret, invalid URL, etc.)
RequestError — network failure or timeout reaching Castle
InvalidRequestTokenError — the request token is missing or invalid
InvalidParametersError — 422 response with validation details
RateLimitError — 429 response; back off and retry
UnauthorizedError — 401; bad API secret
InternalServerError — 5xx response from Castle
WebhookVerificationError — webhook signature did not match
The full list is in castle/errors.py.
Project details
Release history Release notifications | RSS feed
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 castle-7.1.0.tar.gz.
File metadata
- Download URL: castle-7.1.0.tar.gz
- Upload date:
- Size: 19.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
4d4fc8ddbb422173c9df821a291515926afa7896bcd194a47035485e3d6119e1
|
|
| MD5 |
57df564c1ca6dc2df1248a2360ac0d36
|
|
| BLAKE2b-256 |
f8919ac1db7a493f3ea259d6e24daa90e66e85f79659b550503d7e71229a31be
|
File details
Details for the file castle-7.1.0-py3-none-any.whl.
File metadata
- Download URL: castle-7.1.0-py3-none-any.whl
- Upload date:
- Size: 31.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
f9c0935bf585aece8f962f2d3f12d5730739923a7d7a90992c9c246f384f5213
|
|
| MD5 |
ce9ac4369162d8011bdcb82b3000384c
|
|
| BLAKE2b-256 |
fcaa39a6d18a0f9fc499fa9005b604422f3b2b5e3835e662caf5993383fc07c1
|