Metadata-Version: 2.4
Name: aiornot
Version: 0.1.0
Summary: Python client for the AIORNOT API - detect AI-generated content in images, videos, audio, and text.
Project-URL: Homepage, https://aiornot.com/
Project-URL: Repository, https://github.com/aiornotinc/aiornot-python
Project-URL: Bug Tracker, https://github.com/aiornotinc/aiornot-python/issues
Project-URL: Documentation, https://github.com/aiornotinc/aiornot-python
Author-email: aiornot <support@aiornot.com>
License: MIT
License-File: LICENSE
Keywords: ai detection,aiornot,genai,generative ai
Classifier: Development Status :: 4 - Beta
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: MIT License
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Requires-Python: <3.14,>=3.10
Requires-Dist: aiofiles>=23.2.1
Requires-Dist: click>=8.1.7
Requires-Dist: httpx>=0.25.1
Requires-Dist: pydantic>=2.5.2
Description-Content-Type: text/markdown

![AIORNOT Logo](./media/centered_logo_32x32.png)
# AIORNOT Python Client

![Tests](https://github.com/aiornotinc/aiornot-python/actions/workflows/test.yaml/badge.svg)
[![PyPI version](https://badge.fury.io/py/aiornot.svg)](https://badge.fury.io/py/aiornot)
[![Better Stack Badge](https://uptime.betterstack.com/status-badges/v2/monitor/y3x3.svg)](https://uptime.betterstack.com/?utm_source=status_badge)

Python client for the [AIORNOT](https://aiornot.com) API - detect AI-generated content in images, videos, audio, and text.

## Features

- **Image Analysis**: Detect AI-generated images, deepfakes, NSFW content, and more
- **Video Analysis**: Analyze videos for AI-generated content, synthetic voices, and deepfakes
- **Voice Analysis**: Detect AI-generated speech
- **Music Analysis**: Detect AI-generated music
- **Text Analysis**: Detect AI-generated text with block-level annotations
- **Batch Processing**: Process multiple files concurrently with progress tracking
- **Sync & Async**: Both synchronous and asynchronous clients available

## Installation

```bash
# As a CLI tool (no install needed)
uvx aiornot --help

# As a library dependency
uv add aiornot    # or: pip install aiornot
```

Requires Python 3.10+.

## API Key Setup

Register for an account at [AIORNOT](https://aiornot.com) and get your API key from your [dashboard](https://aiornot.com/dashboard/api).

![](./media/api-token-mgmt.png)

Click `Create New Token` to open the dialog:

![](./media/create-token.png)

Copy your token after creating it.

> [!WARNING]
> Never share your API key with anyone. It is like a password.

## CLI Usage

### Token Management

Configure your API key (saved to `~/.aiornot/config.json`):

```bash
aiornot token config
```

Verify your token is valid:

```bash
aiornot token check
```

Alternatively, set an environment variable:

```bash
export AIORNOT_API_KEY=your_api_key
```

### Analyzing Files

```bash
# Analyze an image
aiornot image photo.jpg

# Analyze a video
aiornot video clip.mp4

# Analyze voice/speech audio
aiornot voice speech.mp3

# Analyze music
aiornot music song.wav

# Analyze text (direct input or from file)
aiornot text "Some text to analyze"
aiornot text --file document.txt
```

### Output Formats

```bash
# JSON output (default)
aiornot image photo.jpg --format json

# Human-readable table
aiornot image photo.jpg --format table

# Minimal (verdict + confidence)
aiornot image photo.jpg --format minimal

# Just the verdict
aiornot image photo.jpg --quiet
```

### Analysis Filtering

```bash
# Only run specific analyses
aiornot image photo.jpg --only ai_generated --only deepfake

# Exclude analyses
aiornot image photo.jpg --excluding nsfw --excluding quality
```

### Batch Processing

Process multiple files with JSONL output:

```bash
# Process multiple files
aiornot batch image photo1.jpg photo2.png photo3.webp

# Process a directory
aiornot batch image --dir ./photos
aiornot batch image --dir ./photos --recursive

# Process from CSV file
aiornot batch image --csv files.csv --key image_path
aiornot batch image --csv files.csv --key filename --base-dir /data/images

# Output options
aiornot batch image --dir ./photos --output results.jsonl  # Write to file
aiornot batch image --dir ./photos --format summary        # Summary only
aiornot batch image --dir ./photos --format quiet          # Exit code only

# Control concurrency and progress
aiornot batch image --dir ./photos --concurrency 10 --progress
aiornot batch image --dir ./photos --fail-fast  # Stop on first error
```

Available batch commands: `image`, `video`, `voice`, `music`, `text`

JSONL output format (one JSON object per line):
```jsonl
{"status":"success","input":"photo1.jpg","result":{...}}
{"status":"error","input":"photo2.jpg","error":"FileNotFound","message":"..."}
{"status":"summary","total":2,"succeeded":1,"failed":1,"success_rate":0.5}
```

## Python Library Usage

### Quick Start

```python
from aiornot import Client

# Uses AIORNOT_API_KEY env var or ~/.aiornot/config.json
client = Client()

# Or pass explicitly
client = Client(api_key="your_api_key")

# Analyze an image
resp = client.image_report_from_file("path/to/image.jpg")
print(f"Verdict: {resp.verdict}")      # "ai" or "human"
print(f"Confidence: {resp.confidence}") # 0.0 to 1.0
print(f"Is AI: {resp.is_ai()}")         # True or False

# Analyze text
resp = client.text_report("Some text to analyze for AI generation.")
print(f"Is AI: {resp.is_ai()}")

# Analyze voice audio
resp = client.voice_report_from_file("path/to/speech.mp3")
print(f"Verdict: {resp.verdict}")

# Analyze music
resp = client.music_report_from_file("path/to/song.mp3")
print(f"Verdict: {resp.verdict}")

# Analyze video
resp = client.video_report_from_file("path/to/video.mp4")
print(f"AI Video: {resp.ai_video_detected}")
print(f"AI Voice: {resp.ai_voice_detected}")
```

### Async Client

```python
import asyncio
from aiornot import AsyncClient

async def main():
    client = AsyncClient()

    # All methods are async
    resp = await client.image_report_from_file("image.jpg")
    print(f"Verdict: {resp.verdict}")

asyncio.run(main())
```

### Image Analysis

Control which analyses to run using `only` or `excluding`:

```python
from aiornot import Client, ImageAnalysisType

client = Client()

# Only run AI detection and deepfake analysis
resp = client.image_report_from_file(
    "image.jpg",
    only=[ImageAnalysisType.AI_GENERATED, ImageAnalysisType.DEEPFAKE]
)

# Run all except NSFW detection
resp = client.image_report_from_file(
    "image.jpg",
    excluding=[ImageAnalysisType.NSFW]
)

# Access results
print(f"AI Generated: {resp.is_ai()}")
print(f"Deepfake: {resp.is_deepfake()}")
print(f"NSFW: {resp.is_nsfw()}")

# Generator detection (if AI detected)
if resp.report.ai_generated and resp.report.ai_generated.generator:
    gen = resp.report.ai_generated.generator
    print(f"Midjourney confidence: {gen.midjourney.confidence}")
```

Available `ImageAnalysisType` values:
- `AI_GENERATED` - AI image detection
- `DEEPFAKE` - Face manipulation detection
- `NSFW` - Adult content detection
- `QUALITY` - Image quality assessment
- `REVERSE_SEARCH` - Reverse image search

### Video Analysis

```python
from aiornot import Client, VideoAnalysisType

client = Client()

resp = client.video_report_from_file(
    "video.mp4",
    only=[VideoAnalysisType.AI_VIDEO, VideoAnalysisType.DEEPFAKE_VIDEO]
)

print(f"AI Video: {resp.ai_video_detected} ({resp.ai_video_confidence:.1%})")
print(f"AI Voice: {resp.ai_voice_detected}")
print(f"AI Music: {resp.ai_music_detected}")
print(f"Deepfake: {resp.deepfake_detected}")
```

Available `VideoAnalysisType` values:
- `AI_VIDEO` - AI-generated video detection
- `AI_VOICE` - Synthetic voice detection
- `AI_MUSIC` - AI-generated music detection
- `DEEPFAKE_VIDEO` - Deepfake detection

### Text Analysis

```python
client = Client()

# Basic text analysis
resp = client.text_report("Text to analyze...")
print(f"Is AI: {resp.is_ai()}")
print(f"Confidence: {resp.confidence}")

# With block-level annotations
resp = client.text_report(
    "First paragraph here. Second paragraph here.",
    include_annotations=True
)
for block, confidence in resp.annotations:
    print(f"[{confidence:.1%}] {block[:50]}...")
```

### Batch Processing

Process multiple files concurrently:

```python
from pathlib import Path
from aiornot import Client

client = Client()

# Process multiple images
images = list(Path("images").glob("*.jpg"))
results = client.image_report_batch(
    images,
    max_concurrency=5,
    on_progress=lambda done, total: print(f"{done}/{total}")
)

print(f"Total: {results.total}")
print(f"Succeeded: {results.succeeded}")
print(f"Failed: {results.failed}")
print(f"Success rate: {results.success_rate:.1%}")

# Access individual results
for result in results.results:
    if result.success:
        print(f"{result.input}: {result.result.verdict}")
    else:
        print(f"{result.input}: Error - {result.error}")

# Process a directory
results = client.image_report_directory(
    "path/to/images",
    recursive=True,
    max_concurrency=5
)

# Process from CSV
results = client.image_report_from_csv(
    "files.csv",
    key="file_path",
    base_directory="/data/images"
)
```

## Error Handling

```python
from aiornot import Client
from aiornot.exceptions import (
    AIORNotAuthenticationError,
    AIORNotValidationError,
    AIORNotRateLimitError,
    AIORNotTimeoutError,
)

client = Client()

try:
    resp = client.image_report_from_file("image.jpg")
except AIORNotAuthenticationError:
    print("Invalid API key")
except AIORNotValidationError as e:
    print(f"Invalid input: {e}")
except AIORNotRateLimitError:
    print("Rate limit exceeded")
except AIORNotTimeoutError:
    print("Request timed out")
```

## License

MIT
