Skip to content

TypeScript SDK

Complete API reference for the Vettly TypeScript SDK.

Installation

bash
npm install @nextauralabs/vettly-sdk

Quick Start

typescript
import { ModerationClient } from '@nextauralabs/vettly-sdk'

const client = new ModerationClient({
  apiKey: process.env.VETTLY_API_KEY!
})

const result = await client.check({
  content: 'Text to moderate',
  policyId: 'moderate',
  contentType: 'text'
})

console.log('Safe:', result.safe)

ModerationClient

Constructor

typescript
new ModerationClient(config: ModerationClientConfig)

Config Options

typescript
interface ModerationClientConfig {
  apiKey: string          // Required: Your Vettly API key
  apiUrl?: string         // Optional: API base URL (default: https://api.vettly.dev)
  timeout?: number        // Optional: Request timeout in ms (default: 30000)
  maxRetries?: number     // Optional: Max retries for failed requests (default: 3)
  retryDelay?: number     // Optional: Base delay for exponential backoff in ms (default: 1000)
}

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

Example

typescript
const client = new ModerationClient({
  apiKey: 'sk_live_xxxxx',
  apiUrl: 'https://api.vettly.dev',
  timeout: 30000,     // 30 seconds
  maxRetries: 3,      // Retry up to 3 times
  retryDelay: 1000    // Start with 1s delay
})

Content Moderation

check()

Check content for moderation violations.

typescript
async check(request: CheckRequest): Promise<CheckResponse>

Request

typescript
interface CheckRequest {
  content: string         // Content to check (text, base64 image)
  policyId: string        // Policy ID to use
  contentType: 'text' | 'image' | 'video'
  requestId?: string      // Optional: Idempotency key for deduplication
  metadata?: {            // Optional metadata
    userId?: string
    ip?: string
    userAgent?: string
    [key: string]: any
  }
}

Idempotency: Use requestId to prevent duplicate processing. If you retry the same request with the same requestId, you'll get the cached result instead of processing again.

Response

typescript
interface CheckResponse {
  safe: boolean           // True if content is safe
  flagged: boolean        // True if any category triggered
  action: 'allow' | 'warn' | 'flag' | 'block'
  categories: Category[]  // Category results
  decisionId: string      // Unique decision ID
  provider: string        // AI provider used
  latency: number         // Response time in ms
  cost: number           // Cost in USD
}

interface Category {
  category: string        // Category name
  score: number          // Confidence score (0-1)
  threshold: number      // Policy threshold
  triggered: boolean     // True if score >= threshold
}

Example

typescript
const result = await client.check({
  content: 'Hello, world!',
  policyId: 'moderate',
  contentType: 'text',
  metadata: {
    userId: 'user_123',
    ip: '192.168.1.1'
  }
})

if (result.safe) {
  console.log('Content is safe!')
} else {
  console.log('Flagged categories:', result.categories.filter(c => c.triggered))
}

checkImage()

Convenience method for image moderation.

typescript
async checkImage(
  imageUrl: string,
  options?: {
    policyId?: string
    requestId?: string
    metadata?: Record<string, unknown>
  }
): Promise<CheckResponse>

Example

typescript
// From URL
const result = await client.checkImage(
  'https://example.com/image.jpg',
  { policyId: 'strict' }
)

// From base64
const result = await client.checkImage(
  '...',
  { policyId: 'moderate' }
)

if (result.action === 'block') {
  console.log('Image blocked')
}

dryRun()

Test a policy without making actual AI provider calls.

typescript
async dryRun(
  policyId: string,
  mockScores?: Record<string, number>
): Promise<DryRunResponse>

Example

typescript
const result = await client.dryRun('strict', {
  violence: 0.8,    // Mock: 80% violence score
  sexual: 0.3,      // Mock: 30% sexual score
  hate: 0.95        // Mock: 95% hate score
})

console.log('Would be blocked:', result.action === 'block')
console.log('Triggered categories:', result.categories.filter(c => c.triggered))

batchCheck()

Check multiple items in a single synchronous request.

typescript
async batchCheck(request: {
  policyId: string
  items: Array<{
    id: string
    content: string
    contentType?: 'text' | 'image' | 'video'
    metadata?: Record<string, unknown>
  }>
}): Promise<BatchCheckResponse>

Example

typescript
const results = await client.batchCheck({
  policyId: 'moderate',
  items: [
    { id: 'comment_1', content: 'Great post!' },
    { id: 'comment_2', content: 'Inappropriate content...' },
    { id: 'comment_3', content: 'Thanks for sharing!' }
  ]
})

results.items.forEach(item => {
  console.log(`${item.id}: ${item.safe ? 'Safe' : 'Flagged'}`)
})

batchCheckAsync()

Check multiple items asynchronously with webhook delivery.

typescript
async batchCheckAsync(request: {
  policyId: string
  items: Array<{
    id: string
    content: string
    contentType?: 'text' | 'image' | 'video'
    metadata?: Record<string, unknown>
  }>
  webhookUrl: string
}): Promise<{ batchId: string }>

Example

typescript
const batch = await client.batchCheckAsync({
  policyId: 'moderate',
  items: [
    { id: '1', content: 'Comment 1' },
    { id: '2', content: 'Comment 2' },
    // ... 1000 items
  ],
  webhookUrl: 'https://myapp.com/webhooks/moderation'
})

console.log('Batch ID:', batch.batchId)
// Results will be sent to webhook when ready

Policy Management

createPolicy()

Create or update a moderation policy.

typescript
async createPolicy(
  policyId: string,
  yamlContent: string
): Promise<Policy>

Example

typescript
const yamlContent = `
name: My Custom Policy
categories:
  violence:
    threshold: 0.7
    action: block
  sexual:
    threshold: 0.8
    action: warn
`

const policy = await client.createPolicy(
  'my_custom_policy',
  yamlContent
)

console.log('Policy created:', policy.policyId)

getPolicy()

Get details of a specific policy.

typescript
async getPolicy(policyId: string): Promise<Policy>

Example

typescript
const policy = await client.getPolicy('moderate')

console.log('Policy name:', policy.name)
console.log('Categories:', policy.categories)

listPolicies()

List all available policies.

typescript
async listPolicies(): Promise<{ policies: Policy[] }>

Example

typescript
const { policies } = await client.listPolicies()

policies.forEach(policy => {
  console.log(`${policy.policyId}: ${policy.name}`)
})

Decision Tracking

getDecision()

Get details of a specific moderation decision.

typescript
async getDecision(decisionId: string): Promise<Decision>

Example

typescript
const decision = await client.getDecision('dec_abc123')

console.log('Content:', decision.content)
console.log('Safe:', decision.safe)
console.log('Timestamp:', decision.timestamp)

listDecisions()

List recent moderation decisions.

typescript
async listDecisions(options?: {
  limit?: number
  offset?: number
}): Promise<{ decisions: Decision[] }>

Example

typescript
const { decisions } = await client.listDecisions({
  limit: 100,
  offset: 0
})

decisions.forEach(d => {
  console.log(`${d.decisionId}: ${d.safe ? 'Safe' : 'Flagged'}`)
})

replayDecision()

Replay a past decision with a different policy.

typescript
async replayDecision(
  decisionId: string,
  policyId: string
): Promise<CheckResponse>

Example

typescript
// Original decision used 'moderate' policy
const original = await client.getDecision('dec_abc123')

// Replay with 'strict' policy
const replayed = await client.replayDecision('dec_abc123', 'strict')

console.log('Original action:', original.action)
console.log('Replayed action:', replayed.action)

getCurlCommand()

Get a cURL command to reproduce a decision.

typescript
async getCurlCommand(decisionId: string): Promise<string>

Example

typescript
const curl = await client.getCurlCommand('dec_abc123')

console.log('To reproduce this decision, run:')
console.log(curl)

Output:

bash
curl -X POST https://api.vettly.dev/v1/check \
  -H "Authorization: Bearer vettly_xxxxx" \
  -H "Content-Type: application/json" \
  -d '{"content":"...","policyId":"moderate","contentType":"text"}'

Webhooks

registerWebhook()

Register a webhook endpoint for events.

typescript
async registerWebhook(request: {
  url: string
  events: string[]
  description?: string
}): Promise<Webhook>

Example

typescript
const webhook = await client.registerWebhook({
  url: 'https://myapp.com/webhooks/vettly',
  events: ['moderation.completed', 'moderation.failed'],
  description: 'Production webhook'
})

console.log('Webhook ID:', webhook.id)

listWebhooks()

List all registered webhooks.

typescript
async listWebhooks(): Promise<{ webhooks: Webhook[] }>

Example

typescript
const { webhooks } = await client.listWebhooks()

webhooks.forEach(hook => {
  console.log(`${hook.id}: ${hook.url}`)
})

getWebhook()

Get details of a specific webhook.

typescript
async getWebhook(webhookId: string): Promise<Webhook>

Example

typescript
const webhook = await client.getWebhook('wh_abc123')

console.log('URL:', webhook.url)
console.log('Events:', webhook.events)
console.log('Enabled:', webhook.enabled)

updateWebhook()

Update webhook configuration.

typescript
async updateWebhook(
  webhookId: string,
  updates: {
    url?: string
    events?: string[]
    description?: string
    enabled?: boolean
  }
): Promise<Webhook>

Example

typescript
const updated = await client.updateWebhook('wh_abc123', {
  events: ['moderation.completed'],  // Remove 'failed' event
  enabled: true
})

deleteWebhook()

Delete a webhook endpoint.

typescript
async deleteWebhook(webhookId: string): Promise<void>

Example

typescript
await client.deleteWebhook('wh_abc123')
console.log('Webhook deleted')

testWebhook()

Send a test event to a webhook.

typescript
async testWebhook(
  webhookId: string,
  eventType: string
): Promise<{ success: boolean }>

Example

typescript
const result = await client.testWebhook(
  'wh_abc123',
  'moderation.completed'
)

console.log('Test successful:', result.success)

getWebhookDeliveries()

Get delivery logs for a webhook.

typescript
async getWebhookDeliveries(
  webhookId: string,
  options?: { limit?: number }
): Promise<{ deliveries: Delivery[] }>

Example

typescript
const { deliveries } = await client.getWebhookDeliveries('wh_abc123', {
  limit: 50
})

deliveries.forEach(d => {
  console.log(`${d.timestamp}: ${d.status} (${d.responseCode})`)
})

Webhook Signature Verification

Verify webhook signatures to ensure authenticity:

verifyWebhookSignature()

typescript
function verifyWebhookSignature(
  payload: string,
  signature: string,
  secret: string
): boolean

constructWebhookEvent()

typescript
function constructWebhookEvent(payload: string): WebhookEvent

Example

typescript
import express from 'express'
import {
  verifyWebhookSignature,
  constructWebhookEvent
} from '@nextauralabs/vettly-sdk'

const app = express()

app.post('/webhooks/vettly',
  express.raw({ type: 'application/json' }),
  (req, res) => {
    const payload = req.body.toString()
    const signature = req.headers['x-vettly-signature'] as string
    const webhookSecret = process.env.VETTLY_WEBHOOK_SECRET!

    if (!verifyWebhookSignature(payload, signature, webhookSecret)) {
      return res.status(401).send('Invalid signature')
    }

    const event = constructWebhookEvent(payload)

    switch (event.type) {
      case 'decision.blocked':
        console.log('Content blocked:', event.data.id)
        break
      case 'decision.flagged':
        console.log('Content flagged:', event.data.id)
        break
    }

    res.status(200).send('OK')
  }
)

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

Express Middleware

moderateContent()

Express.js middleware for automatic content moderation.

typescript
function moderateContent(options: {
  client: ModerationClient
  policyId: string
  field?: string
  onFlagged?: (req, res, result: CheckResponse) => void
}): ExpressMiddleware

Example

typescript
import express from 'express'
import { ModerationClient, moderateContent } from '@nextauralabs/vettly-sdk'

const app = express()
const client = new ModerationClient({ apiKey: 'vettly_xxxxx' })

app.post('/api/comments',
  moderateContent({
    client,
    policyId: 'moderate',
    field: 'body.content',  // Check req.body.content
    onFlagged: (req, res, result) => {
      // Custom handling
      res.status(400).json({
        error: 'Content flagged',
        categories: result.categories.filter(c => c.triggered)
      })
    }
  }),
  async (req, res) => {
    // Only reaches here if content is safe
    const comment = await saveComment(req.body)
    res.json(comment)
  }
)

Default Behavior

If onFlagged is not provided:

  • action: 'block' → 403 response with error
  • action: 'warn' or 'flag' → Continue to next middleware
  • action: 'allow' → Continue to next middleware

Nested Fields

typescript
app.post('/api/posts',
  moderateContent({
    client,
    policyId: 'moderate',
    field: 'body.post.content'  // Checks req.body.post.content
  }),
  handler
)

Error Handling

The SDK provides typed exceptions for better error handling:

typescript
import {
  ModerationClient,
  VettlyAuthError,
  VettlyRateLimitError,
  VettlyQuotaError,
  VettlyValidationError,
  VettlyError,
} from '@nextauralabs/vettly-sdk'

const client = new ModerationClient({ apiKey: 'sk_live_...' })

try {
  const result = await client.check({
    content: 'Text',
    policyId: 'moderate',
    contentType: 'text'
  })
} catch (error) {
  if (error instanceof VettlyAuthError) {
    console.error('Invalid API key')
  } else if (error instanceof VettlyRateLimitError) {
    console.error(`Rate limited. Retry after ${error.retryAfter}s`)
  } else if (error instanceof VettlyQuotaError) {
    console.error(`Quota exceeded: ${error.quota}`)
  } else if (error instanceof VettlyValidationError) {
    console.error(`Invalid request: ${error.errors}`)
  } else if (error instanceof VettlyError) {
    console.error(`API error: ${error.message}`)
  }
}

Error Types

Error ClassHTTP StatusDescription
VettlyAuthError401Invalid or expired API key
VettlyValidationError422Invalid request parameters
VettlyRateLimitError429Too many requests (has retryAfter)
VettlyQuotaError402Usage quota exceeded (has quota)
VettlyError5xxServer error (auto-retried by SDK)

Fail Open Strategy

For production, decide how to handle moderation failures:

typescript
async function moderateContent(content: string): Promise<boolean> {
  try {
    const result = await client.check({
      content,
      policyId: 'moderate',
      contentType: 'text'
    })
    return result.action !== 'block'
  } catch (error) {
    // Fail open: allow content if moderation fails
    console.error('Moderation failed:', error)
    return true
  }
}

TypeScript Types

All types are exported from the SDK:

typescript
import type {
  // Request/Response
  CheckRequest,
  CheckResponse,
  Category,
  ModerationClientConfig,
  // Resources
  Policy,
  Decision,
  Webhook,
  WebhookEvent,
} from '@nextauralabs/vettly-sdk'

// Error classes (not types, actual classes)
import {
  VettlyError,
  VettlyAuthError,
  VettlyRateLimitError,
  VettlyQuotaError,
  VettlyValidationError,
} from '@nextauralabs/vettly-sdk'

const request: CheckRequest = {
  content: 'Text to check',
  policyId: 'moderate',
  contentType: 'text'
}

const handleResult = (response: CheckResponse) => {
  console.log('Safe:', response.safe)
}

Advanced Usage

Rate Limiting

typescript
import pLimit from 'p-limit'

const limit = pLimit(10) // Max 10 concurrent requests

const results = await Promise.all(
  comments.map(comment =>
    limit(() =>
      client.check({
        content: comment.text,
        policyId: 'moderate',
        contentType: 'text'
      })
    )
  )
)

Caching Results

typescript
const cache = new Map<string, CheckResponse>()

async function checkWithCache(
  client: ModerationClient,
  content: string,
  policyId: string
): Promise<CheckResponse> {
  const cacheKey = `${policyId}:${content}`

  if (cache.has(cacheKey)) {
    return cache.get(cacheKey)!
  }

  const result = await client.check({
    content,
    policyId,
    contentType: 'text'
  })

  cache.set(cacheKey, result)
  return result
}

Multi-Policy Check

typescript
async function checkMultiplePolicies(
  client: ModerationClient,
  content: string,
  policyIds: string[]
): Promise<Map<string, CheckResponse>> {
  const results = await Promise.all(
    policyIds.map(policyId =>
      client.check({ content, policyId, contentType: 'text' })
    )
  )

  return new Map(
    policyIds.map((policyId, i) => [policyId, results[i]])
  )
}

const results = await checkMultiplePolicies(
  client,
  'Hello world',
  ['lenient', 'moderate', 'strict']
)

console.log('Lenient:', results.get('lenient')?.safe)
console.log('Moderate:', results.get('moderate')?.safe)
console.log('Strict:', results.get('strict')?.safe)

See Also