Self-hosted content moderation service powered by OpenAI's moderation API
GateKeeper is a Python-based content moderation service that uses FastAPI to provide a high-performance API for moderating text and images. It forwards content to OpenAI's moderation API and returns the moderation result with a cryptographic signature for verification.
Create a .env
file based on the .env.example
template:
cp .env.example .env
Then edit the .env
file to set your OpenAI API key, signature secret, and security parameters:
# Required settings
OPENAI_API_KEY=your_openai_api_key_here
GATEKEEPER_SIGNATURE_SECRET=your_signature_secret_here
OPENAI_MODERATION_MODEL=text-moderation-latest
# Vision settings
GATEKEEPER_ENABLE_VISION=false
GATEKEEPER_VISION_MODEL=gpt-4o-mini
GATEKEEPER_VISION_DETAIL=auto
GATEKEEPER_VISION_PROMPT="Analyze this image for a restaurant review or venue listing..."
# Security settings
GATEKEEPER_ALLOWED_ORIGINS=https://yourdomain.com,https://anotherdomain.com
GATEKEEPER_API_KEYS=key1,key2,key3
GATEKEEPER_MAX_TEXT_LENGTH=10000
GATEKEEPER_MAX_IMAGE_SIZE=5242880
GATEKEEPER_MAX_ITEMS=10
GATEKEEPER_RATE_LIMIT_TIMES=10
GATEKEEPER_RATE_LIMIT_SECONDS=60
Install dependencies:
pip install -r requirements.txt
Run the server:
uvicorn app:app --reload
The API will be available at http://localhost:8000
Build the Docker image:
docker build -t gatekeeper .
Run the container:
docker run -p 8000:8000 --env-file .env gatekeeper
When API key authentication is enabled, include your API key in the X-API-Key
header:
X-API-Key: your-api-key-here
The API enforces rate limiting based on client IP address. By default, this is set to 10 requests per 60 seconds, but can be configured via environment variables. When rate limits are exceeded, the API returns a 429 Too Many Requests
response.
The API enforces limits on:
Endpoint: POST /moderate
The API supports flexible input combinations:
Request Example with Multiple Items:
{
"input": [
{ "type": "text", "text": "This restaurant has amazing food!" },
{ "type": "text", "text": "The service was excellent and I highly recommend the steak." },
{ "type": "image", "base64": "..." },
{ "type": "image", "base64": "..." }
],
"analyze_images": true
}
The analyze_images
parameter is optional and defaults to false
. If set to true
and vision analysis is enabled in the server configuration, images will be analyzed using OpenAI's vision models after moderation.
Response Example with Multiple Items:
{
"id": "modr-123456",
"model": "text-moderation-latest",
"results": [
{
"flagged": false,
"categories": {
"hate": false,
"harassment": false,
"self-harm": false,
"sexual": false,
"violence": false
},
"category_scores": {
"hate": 0.0,
"harassment": 0.0,
"self-harm": 0.0,
"sexual": 0.0,
"violence": 0.0
}
},
{
"flagged": false,
"categories": {
"hate": false,
"harassment": false,
"self-harm": false,
"sexual": false,
"violence": false
},
"category_scores": {
"hate": 0.0,
"harassment": 0.0,
"self-harm": 0.0,
"sexual": 0.0,
"violence": 0.0
}
},
{
"flagged": false,
"categories": {
"hate": false,
"harassment": false,
"self-harm": false,
"sexual": false,
"violence": false
},
"category_scores": {
"hate": 0.0,
"harassment": 0.0,
"self-harm": 0.0,
"sexual": 0.0,
"violence": 0.0
}
},
{
"flagged": false,
"categories": {
"hate": false,
"harassment": false,
"self-harm": false,
"sexual": false,
"violence": false
},
"category_scores": {
"hate": 0.0,
"harassment": 0.0,
"self-harm": 0.0,
"sexual": 0.0,
"violence": 0.0
}
}
],
"combined_text_moderation": {
"flagged": false,
"categories": {
"hate": false,
"harassment": false,
"self-harm": false,
"sexual": false,
"violence": false
},
"category_scores": {
"hate": 0.0,
"harassment": 0.0,
"self-harm": 0.0,
"sexual": 0.0,
"violence": 0.0
}
},
"vision_analysis": [
{
"image_index": 2,
"analysis": "This image shows a plate of beautifully presented food in what appears to be a restaurant setting. The dish looks like a well-plated steak with garnishes and side dishes. The lighting is good, the food appears appetizing, and the image is of high quality. This is perfectly appropriate for a restaurant listing or review as it showcases the food offerings. VERDICT: APPROPRIATE",
"flagged": false
},
{
"image_index": 3,
"analysis": "This image shows the interior of a restaurant with tables and chairs. The space appears clean, well-lit, and professionally designed. This is an appropriate image for a restaurant listing as it accurately represents the dining environment for potential customers. There are no inappropriate elements or overlays in the image. VERDICT: APPROPRIATE",
"flagged": false
}
],
"flagged": false,
"signature": "abcdef1234567890"
}
Response Fields:
flagged
: Top-level boolean indicating if any content (individual texts, combined text, or images) was flagged as inappropriateresults
: OpenAI moderation results for each individual input itemcombined_text_moderation
: Moderation result from evaluating all text items together as a single input (only relevant when multiple text items are present)vision_analysis
: (Only if requested) Analysis of images, including:image_index
: The index of the image in the input arrayanalysis
: The text analysis of the image contentflagged
: Boolean indicating if the image was flagged as inappropriatesignature
: HMAC-SHA256 signature of the response for verificationThe service uses a comprehensive approach for text moderation:
This dual-layered approach helps identify complex issues such as:
The combined_text_moderation
field only contains meaningful data when there are multiple text items in the request. For single text items or requests with no text, this field will exist but won't provide additional insights beyond the individual item moderation.
The image analysis feature can be configured with these environment variables:
Variable | Description | Default |
---|---|---|
GATEKEEPER_ENABLE_VISION | Enable/disable vision analysis | false |
GATEKEEPER_VISION_MODEL | OpenAI model to use for vision | gpt-4o-mini |
GATEKEEPER_VISION_DETAIL | Detail level for image analysis (low/high/auto) | auto |
GATEKEEPER_VISION_PROMPT | Custom prompt for image analysis | "Analyze this image for a restaurant review..." |
GateKeeper allows you to set custom thresholds for each content category to override OpenAI's default flagging behavior. By default, OpenAI only flags content when it's highly confident the content violates their policies. You can adjust these thresholds to be more or less sensitive depending on your needs.
Set these thresholds using environment variables with the pattern GATEKEEPER_THRESHOLD_<CATEGORY>
where <CATEGORY>
is the uppercase name of the category (e.g., HARASSMENT, HATE, SEXUAL, etc.).
Examples:
GATEKEEPER_THRESHOLD_HARASSMENT=0.1
GATEKEEPER_THRESHOLD_HATE=0.1
GATEKEEPER_THRESHOLD_SEXUAL=0.3
GATEKEEPER_THRESHOLD_VIOLENCE=0.2
For convenience, you can also set a global threshold that applies to all categories using:
GATEKEEPER_THRESHOLD_ALL=0.2
When using GATEKEEPER_THRESHOLD_ALL
, this threshold will be applied to any category that doesn't have a specific threshold set. Category-specific thresholds will always take precedence over the global threshold.
The threshold values range from 0.0 to 1.0:
When a category score from OpenAI exceeds your specified threshold, GateKeeper will flag the content even if OpenAI's default classification did not flag it. This allows for fine-tuning the sensitivity of moderation based on your specific requirements.
The API returns appropriate HTTP status codes for different error scenarios:
400 Bad Request
: Invalid request format403 Forbidden
: Invalid or missing API key429 Too Many Requests
: Rate limit exceeded500 Internal Server Error
: Server-side errorsError responses have a consistent format:
{
"detail": "Error message"
}
The service logs security events to both the console and a security.log
file. Log entries include:
To verify that a response came from the GateKeeper service, recompute the signature:
import hmac
import hashlib
import json
def verify_signature(response, secret):
# Create a copy of the response without the signature
payload = response.copy()
received_signature = payload.pop("signature", None)
# Compute the signature
payload_str = json.dumps(payload, sort_keys=True)
expected_signature = hmac.new(
secret.encode("utf-8"),
payload_str.encode("utf-8"),
hashlib.sha256
).hexdigest()
# Compare signatures
return hmac.compare_digest(expected_signature, received_signature)
API documentation is available at http://localhost:8000/docs
when the server is running.