Skip to content

Python Integration

Add content moderation to your Python app with a simple SDK.

Installation

bash
pip install vettly

Quick Start

python
from vettly import ModerationClient

client = ModerationClient(api_key="sk_live_...")

result = client.check(
    content="Hello, this is a friendly message!",
    policy_id="moderate"
)

if result.action == "block":
    print("Content blocked:", result.categories)
else:
    print("Content allowed")

Configuration

python
from vettly import ModerationClient

client = ModerationClient(
    api_key="sk_live_...",
    api_url="https://api.vettly.dev",  # Optional: custom API URL
    timeout=30.0,                       # Request timeout in seconds (default: 30)
    max_retries=3,                      # Max retries for failures (default: 3)
    retry_delay=1.0,                    # Base delay for backoff in seconds (default: 1)
)

The SDK automatically retries failed requests with exponential backoff:

  • Retries on rate limits (429) and server errors (5xx)
  • Uses exponential backoff (1s, 2s, 4s, ...)
  • Respects Retry-After headers from the API

Text Moderation

python
result = client.check(
    content="User-generated text",
    policy_id="moderate"
)

print(result.action)      # 'allow' | 'flag' | 'block'
print(result.safe)        # True if content is safe
print(result.flagged)     # True if any category triggered
print(result.id)          # Decision ID for audit trail

for category in result.categories:
    print(f"{category.category}: {category.score} (triggered: {category.triggered})")

Image Moderation

Use the convenience method for images:

python
# From URL
result = client.check_image(
    image_url="https://example.com/image.jpg",
    policy_id="strict"
)

# From base64
result = client.check_image(
    image_url="...",
    policy_id="strict"
)

if result.action == "block":
    print("Image blocked")

Or use the general check() method:

python
import base64

with open("image.jpg", "rb") as f:
    image_base64 = base64.b64encode(f.read()).decode()

result = client.check(
    content=f"data:image/jpeg;base64,{image_base64}",
    policy_id="strict",
    content_type="image"
)

Idempotency

Prevent duplicate processing with request IDs:

python
result = client.check(
    content="Hello",
    policy_id="default",
    request_id="unique-request-id-123"  # Idempotency key
)

If you retry the same request with the same request_id, you'll get the cached result instead of processing again.

Error Handling

The SDK provides typed exceptions for better error handling:

python
from vettly import (
    ModerationClient,
    VettlyAuthError,
    VettlyRateLimitError,
    VettlyQuotaError,
    VettlyValidationError,
    VettlyServerError,
)

client = ModerationClient(api_key="sk_live_...")

try:
    result = client.check(content="test", policy_id="default")
except VettlyAuthError:
    print("Invalid API key")
except VettlyRateLimitError as e:
    print(f"Rate limited. Retry after {e.retry_after}s")
except VettlyQuotaError as e:
    print(f"Quota exceeded: {e.quota}")
except VettlyValidationError as e:
    print(f"Invalid request: {e.errors}")
except VettlyServerError:
    print("Server error - retry later")

Fail Open Strategy

For production, decide how to handle moderation failures:

python
def moderate_content(content: str) -> bool:
    """Returns True if content should be allowed."""
    try:
        result = client.check(content=content, policy_id="moderate")
        return result.action != "block"
    except Exception as e:
        # Fail open: allow content if moderation fails
        print(f"Moderation failed: {e}")
        return True

Webhook Signature Verification

Verify webhook signatures to ensure authenticity:

python
from vettly import verify_webhook_signature, construct_webhook_event

# Flask example
@app.route("/webhooks/vettly", methods=["POST"])
def handle_webhook():
    payload = request.get_data(as_text=True)
    signature = request.headers.get("X-Vettly-Signature")
    webhook_secret = "whsec_..."  # From dashboard

    if not verify_webhook_signature(payload, signature, webhook_secret):
        return "Invalid signature", 401

    event = construct_webhook_event(payload)

    if event["type"] == "decision.blocked":
        decision = event["data"]
        print(f"Content blocked: {decision['id']}")

    return "OK", 200

The signature format is t=timestamp,v1=signature. The SDK:

  • Validates the timestamp is within 5 minutes
  • Uses constant-time comparison to prevent timing attacks

Async Support

For async applications, use AsyncModerationClient:

python
from vettly import AsyncModerationClient

async def moderate_content():
    async with AsyncModerationClient(api_key="sk_live_...") as client:
        result = await client.check(
            content="Check this message",
            policy_id="moderate"
        )
        return result.action

# Or without context manager
client = AsyncModerationClient(api_key="sk_live_...")
result = await client.check(content="Hello", policy_id="moderate")
await client.close()

Flask Example

python
from flask import Flask, request, jsonify
from vettly import ModerationClient, VettlyError

app = Flask(__name__)
client = ModerationClient(api_key="sk_live_...")

@app.route('/api/comments', methods=['POST'])
def create_comment():
    data = request.json
    content = data.get('content')

    try:
        result = client.check(content=content, policy_id="moderate")

        if result.action == "block":
            return jsonify({
                "error": "Content blocked",
                "categories": [c.category for c in result.categories if c.triggered]
            }), 403

        # Save to database
        return jsonify({"success": True})

    except VettlyError as e:
        # Fail open on moderation errors
        print(f"Moderation error: {e}")
        return jsonify({"success": True})

if __name__ == '__main__':
    app.run()

FastAPI Example

python
from fastapi import FastAPI, HTTPException
from pydantic import BaseModel
from vettly import AsyncModerationClient, VettlyError

app = FastAPI()
client = AsyncModerationClient(api_key="sk_live_...")

class Comment(BaseModel):
    content: str

@app.post("/api/comments")
async def create_comment(comment: Comment):
    try:
        result = await client.check(
            content=comment.content,
            policy_id="moderate"
        )

        if result.action == "block":
            raise HTTPException(
                status_code=403,
                detail="Content blocked by moderation"
            )

        return {"success": True}

    except VettlyError:
        # Fail open
        return {"success": True}

@app.on_event("shutdown")
async def shutdown():
    await client.close()

Django Example

python
# views.py
from django.http import JsonResponse
from django.views.decorators.http import require_POST
from django.views.decorators.csrf import csrf_exempt
from vettly import ModerationClient, VettlyError
import json

client = ModerationClient(api_key="sk_live_...")

@csrf_exempt
@require_POST
def create_comment(request):
    data = json.loads(request.body)
    content = data.get('content')

    try:
        result = client.check(content=content, policy_id="moderate")

        if result.action == "block":
            return JsonResponse({"error": "Content blocked"}, status=403)

        # Save comment
        return JsonResponse({"success": True})

    except VettlyError as e:
        # Fail open
        return JsonResponse({"success": True})

Batch Moderation

For multiple items, use the batch endpoint:

python
from vettly import ModerationClient, BatchItem

client = ModerationClient(api_key="sk_live_...")

items = [
    BatchItem(id="1", content="First message"),
    BatchItem(id="2", content="Second message"),
    BatchItem(id="3", content="Third message"),
]

result = client.batch_check(policy_id="moderate", items=items)

print(f"Batch ID: {result.batch_id}")
print(f"Total: {result.total}, Completed: {result.completed}")

for item_result in result.results:
    print(f"{item_result.id}: {item_result.action}")

Custom Policies

Use your custom policies by ID:

python
result = client.check(
    content="Message to check",
    policy_id="my-custom-policy"  # Your policy ID from dashboard
)

Built-in policies: lenient, moderate, strict

Response Structure

python
result = client.check(content="Hello", policy_id="moderate")

# CheckResponse attributes
result.id           # str: Unique decision ID
result.safe         # bool: True if content is safe
result.flagged      # bool: True if any category triggered
result.action       # Action: 'allow' | 'flag' | 'block'
result.categories   # List[CategoryResult]: Category scores
result.latency_ms   # int: Response time in milliseconds
result.policy_id    # str: Policy used
result.provider     # str: AI provider used (e.g., 'hive')

# CategoryResult attributes
for cat in result.categories:
    cat.category    # str: Category name
    cat.score       # float: Score 0-1
    cat.threshold   # float: Threshold for triggering
    cat.triggered   # bool: Whether threshold was exceeded

Environment Variables

bash
export VETTLY_API_KEY=sk_live_...
python
import os
from vettly import ModerationClient

client = ModerationClient(api_key=os.environ["VETTLY_API_KEY"])

Next Steps