Skip to main content

Castle protects your users from account compromise

Project description

Build Status

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


Download files

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

Source Distribution

castle-7.1.0.tar.gz (19.5 kB view details)

Uploaded Source

Built Distribution

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

castle-7.1.0-py3-none-any.whl (31.4 kB view details)

Uploaded Python 3

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

Hashes for castle-7.1.0.tar.gz
Algorithm Hash digest
SHA256 4d4fc8ddbb422173c9df821a291515926afa7896bcd194a47035485e3d6119e1
MD5 57df564c1ca6dc2df1248a2360ac0d36
BLAKE2b-256 f8919ac1db7a493f3ea259d6e24daa90e66e85f79659b550503d7e71229a31be

See more details on using hashes here.

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

Hashes for castle-7.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 f9c0935bf585aece8f962f2d3f12d5730739923a7d7a90992c9c246f384f5213
MD5 ce9ac4369162d8011bdcb82b3000384c
BLAKE2b-256 fcaa39a6d18a0f9fc499fa9005b604422f3b2b5e3835e662caf5993383fc07c1

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