documenso-sdk-patterns
Apply production-ready Documenso SDK patterns for TypeScript and Python. Use when implementing Documenso integrations, refactoring SDK usage, or establishing team coding standards for Documenso. Trigger with phrases like "documenso SDK patterns", "documenso best practices", "documenso code patterns", "idiomatic documenso". allowed-tools: Read, Write, Edit version: 1.0.0 license: MIT author: Jeremy Longshore <jeremy@intentsolutions.io>
Allowed Tools
No tools specified
Provided by Plugin
documenso-pack
Claude Code skill pack for Documenso (24 skills)
Installation
This skill is included in the documenso-pack plugin:
/plugin install documenso-pack@claude-code-plugins-plus
Click to copy
Instructions
# Documenso SDK Patterns
## Overview
Production-ready patterns for Documenso SDK usage in TypeScript and Python.
## Prerequisites
- Completed `documenso-install-auth` setup
- Familiarity with async/await patterns
- Understanding of error handling best practices
## Instructions
### Step 1: Singleton Client Pattern
```typescript
// src/documenso/client.ts
import { Documenso } from "@documenso/sdk-typescript";
let instance: Documenso | null = null;
export interface DocumensoClientConfig {
apiKey?: string;
baseUrl?: string;
timeout?: number;
}
export function getDocumensoClient(config?: DocumensoClientConfig): Documenso {
if (!instance) {
const apiKey = config?.apiKey ?? process.env.DOCUMENSO_API_KEY;
if (!apiKey) {
throw new Error(
"DOCUMENSO_API_KEY environment variable is required"
);
}
instance = new Documenso({
apiKey,
serverURL: config?.baseUrl ?? process.env.DOCUMENSO_BASE_URL,
timeoutMs: config?.timeout ?? 30000,
});
}
return instance;
}
// For testing - allows resetting the singleton
export function resetDocumensoClient(): void {
instance = null;
}
```
### Step 2: Type-Safe Error Handling
```typescript
// src/documenso/errors.ts
import { SDKError } from "@documenso/sdk-typescript";
export class DocumensoError extends Error {
constructor(
message: string,
public readonly code: string,
public readonly statusCode: number,
public readonly retryable: boolean,
public readonly originalError?: Error
) {
super(message);
this.name = "DocumensoError";
}
}
export function wrapDocumensoError(error: unknown): DocumensoError {
if (error instanceof SDKError) {
const retryable = error.statusCode >= 500 || error.statusCode === 429;
return new DocumensoError(
error.message,
`DOCUMENSO_${error.statusCode}`,
error.statusCode,
retryable,
error
);
}
if (error instanceof Error) {
return new DocumensoError(
error.message,
"DOCUMENSO_UNKNOWN",
0,
false,
error
);
}
return new DocumensoError(
String(error),
"DOCUMENSO_UNKNOWN",
0,
false
);
}
// Safe wrapper for API calls
export async function safeDocumensoCall(
operation: () => Promise
): Promise<{ data: T | null; error: DocumensoError | null }> {
try {
const data = await operation();
return { data, error: null };
} catch (err) {
const error = wrapDocumensoError(err);
console.error({
code: error.code,
message: error.message,
statusCode: error.statusCode,
retryable: error.retryable,
});
return { data: null, error };
}
}
```
### Step 3: Retry Logic with Exponential Backoff
```typescript
// src/documenso/retry.ts
interface RetryConfig {
maxRetries?: number;
baseDelayMs?: number;
maxDelayMs?: number;
jitterMs?: number;
}
const DEFAULT_RETRY_CONFIG: Required = {
maxRetries: 3,
baseDelayMs: 1000,
maxDelayMs: 30000,
jitterMs: 500,
};
export async function withRetry(
operation: () => Promise,
config: RetryConfig = {}
): Promise {
const { maxRetries, baseDelayMs, maxDelayMs, jitterMs } = {
...DEFAULT_RETRY_CONFIG,
...config,
};
for (let attempt = 0; attempt <= maxRetries; attempt++) {
try {
return await operation();
} catch (error: any) {
const statusCode = error.statusCode ?? error.status ?? 0;
// Only retry on 429 (rate limit) and 5xx (server errors)
const isRetryable = statusCode === 429 || statusCode >= 500;
if (attempt === maxRetries || !isRetryable) {
throw error;
}
const exponentialDelay = baseDelayMs * Math.pow(2, attempt);
const jitter = Math.random() * jitterMs;
const delay = Math.min(exponentialDelay + jitter, maxDelayMs);
console.log(
`Attempt ${attempt + 1} failed. Retrying in ${delay.toFixed(0)}ms...`
);
await new Promise((r) => setTimeout(r, delay));
}
}
throw new Error("Unreachable");
}
```
### Step 4: Document Service Facade
```typescript
// src/services/document-service.ts
import { getDocumensoClient } from "../documenso/client";
import { withRetry } from "../documenso/retry";
import { safeDocumensoCall, DocumensoError } from "../documenso/errors";
export interface CreateDocumentInput {
title: string;
file?: Blob;
recipients: Array<{
email: string;
name: string;
role?: "SIGNER" | "VIEWER" | "APPROVER";
}>;
fields?: Array<{
recipientIndex: number;
type: "SIGNATURE" | "INITIALS" | "NAME" | "EMAIL" | "DATE" | "TEXT";
page: number;
x: number;
y: number;
width?: number;
height?: number;
}>;
sendImmediately?: boolean;
}
export interface DocumentResult {
documentId: string;
status: string;
recipients: Array<{ id: string; email: string }>;
}
export class DocumentService {
private client = getDocumensoClient();
async createDocument(input: CreateDocumentInput): Promise {
// Step 1: Create document
const doc = await withRetry(() =>
this.client.documents.createV0({
title: input.title,
file: input.file,
})
);
const documentId = doc.documentId!;
const recipientIds: Array<{ id: string; email: string }> = [];
// Step 2: Add recipients
for (const recipient of input.recipients) {
const result = await withRetry(() =>
this.client.documentsRecipients.createV0({
documentId,
email: recipient.email,
name: recipient.name,
role: recipient.role ?? "SIGNER",
})
);
recipientIds.push({
id: result.recipientId!,
email: recipient.email,
});
}
// Step 3: Add fields
if (input.fields) {
for (const field of input.fields) {
const recipientId = recipientIds[field.recipientIndex]?.id;
if (!recipientId) continue;
await withRetry(() =>
this.client.documentsFields.createV0({
documentId,
recipientId,
type: field.type,
page: field.page,
positionX: field.x,
positionY: field.y,
width: field.width ?? 200,
height: field.height ?? 60,
})
);
}
}
// Step 4: Send if requested
if (input.sendImmediately) {
await withRetry(() =>
this.client.documents.sendV0({ documentId })
);
}
return {
documentId,
status: input.sendImmediately ? "SENT" : "DRAFT",
recipients: recipientIds,
};
}
async getDocument(documentId: string): Promise {
const { data, error } = await safeDocumensoCall(() =>
this.client.documents.getV0({ documentId })
);
if (error) {
if (error.statusCode === 404) return null;
throw error;
}
return {
documentId: data!.id!,
status: data!.status!,
recipients:
data!.recipients?.map((r) => ({
id: r.id!,
email: r.email!,
})) ?? [],
};
}
async deleteDocument(documentId: string): Promise {
const { error } = await safeDocumensoCall(() =>
this.client.documents.deleteV0({ documentId })
);
return error === null;
}
}
// Singleton instance
let documentService: DocumentService | null = null;
export function getDocumentService(): DocumentService {
if (!documentService) {
documentService = new DocumentService();
}
return documentService;
}
```
### Step 5: Response Validation with Zod
```typescript
// src/documenso/types.ts
import { z } from "zod";
export const DocumentStatusSchema = z.enum([
"DRAFT",
"PENDING",
"COMPLETED",
"REJECTED",
"CANCELLED",
]);
export const RecipientSchema = z.object({
id: z.string(),
email: z.string().email(),
name: z.string(),
role: z.enum(["SIGNER", "VIEWER", "APPROVER", "CC"]),
signingStatus: z.enum(["NOT_SIGNED", "SIGNED", "REJECTED"]).optional(),
});
export const DocumentSchema = z.object({
id: z.string(),
title: z.string(),
status: DocumentStatusSchema,
createdAt: z.string().datetime(),
updatedAt: z.string().datetime(),
recipients: z.array(RecipientSchema).optional(),
});
export type Document = z.infer;
export type DocumentStatus = z.infer;
// Validate API response
export function validateDocument(data: unknown): Document {
return DocumentSchema.parse(data);
}
```
### Python Patterns
```python
# src/documenso/client.py
import os
from typing import Optional
from functools import lru_cache
from documenso_sdk import Documenso
@lru_cache(maxsize=1)
def get_documenso_client(
api_key: Optional[str] = None,
base_url: Optional[str] = None,
) -> Documenso:
"""Get singleton Documenso client."""
key = api_key or os.environ.get("DOCUMENSO_API_KEY")
if not key:
raise ValueError("DOCUMENSO_API_KEY is required")
return Documenso(
api_key=key,
server_url=base_url or os.environ.get("DOCUMENSO_BASE_URL"),
)
# src/documenso/retry.py
import asyncio
import random
from typing import TypeVar, Callable, Awaitable
T = TypeVar("T")
async def with_retry(
operation: Callable[[], Awaitable[T]],
max_retries: int = 3,
base_delay: float = 1.0,
max_delay: float = 30.0,
) -> T:
"""Execute operation with exponential backoff retry."""
for attempt in range(max_retries + 1):
try:
return await operation()
except Exception as e:
status = getattr(e, "status_code", 0)
is_retryable = status == 429 or status >= 500
if attempt == max_retries or not is_retryable:
raise
delay = min(base_delay * (2 ** attempt) + random.uniform(0, 0.5), max_delay)
print(f"Attempt {attempt + 1} failed. Retrying in {delay:.1f}s...")
await asyncio.sleep(delay)
raise RuntimeError("Unreachable")
```
## Output
- Type-safe client singleton
- Robust error handling with retryable classification
- Automatic retry with exponential backoff
- Service facade for common operations
- Runtime validation for API responses
## Error Handling Patterns
| Pattern | Use Case | Benefit |
|---------|----------|---------|
| Safe wrapper | All API calls | Prevents uncaught exceptions |
| Retry logic | Transient failures | Improves reliability |
| Type guards | Response validation | Catches API changes |
| Service facade | Complex workflows | Encapsulates multi-step operations |
## Resources
- [Documenso SDK Documentation](https://github.com/documenso/sdk-typescript)
- [Zod Documentation](https://zod.dev/)
- [Error Handling Best Practices](https://docs.documenso.com/developers)
## Next Steps
Apply patterns in `documenso-core-workflow-a` for document creation workflows.