@devwithbobby/loops
v0.2.0
Published
Convex component for integrating with Loops.so email marketing platform
Maintainers
Readme
@devwithbobby/loops
A Convex component for integrating with Loops.so email marketing platform. Send transactional emails, manage contacts, trigger loops, and monitor email operations with built-in spam detection and rate limiting.
Features
- Contact Management - Create, update, find, list, and delete contacts
- Transactional Emails - Send one-off emails with templates
- Events - Trigger email workflows based on events
- Loops - Trigger automated email sequences
- Monitoring - Track all email operations with spam detection
- Rate Limiting - Built-in rate limiting queries for abuse prevention
- Type-Safe - Full TypeScript support with Zod validation
Installation
npm install @devwithbobby/loops
# or
bun add @devwithbobby/loopsQuick Start
1. Install and Mount the Component
In your convex/convex.config.ts:
import loops from "@devwithbobby/loops/convex.config";
import { defineApp } from "convex/server";
const app = defineApp();
app.use(loops);
export default app;2. Set Up Environment Variables
IMPORTANT: Set your Loops API key before using the component.
npx convex env set LOOPS_API_KEY "your-loops-api-key-here"Or via Convex Dashboard:
- Go to Settings -> Environment Variables
- Add
LOOPS_API_KEYwith your Loops.so API key
Get your API key from Loops.so Dashboard.
3. Use the Component
In your convex/functions.ts (or any convex file):
import { Loops } from "@devwithbobby/loops";
import { components } from "./_generated/api";
import { action } from "./_generated/server";
import { v } from "convex/values";
// Initialize the Loops client
const loops = new Loops(components.loops);
// Export functions wrapped with auth (required in production)
export const addContact = action({
args: {
email: v.string(),
firstName: v.optional(v.string()),
lastName: v.optional(v.string()),
},
handler: async (ctx, args) => {
// Add authentication check
const identity = await ctx.auth.getUserIdentity();
if (!identity) throw new Error("Unauthorized");
return await loops.addContact(ctx, args);
},
});
export const sendWelcomeEmail = action({
args: {
email: v.string(),
name: v.string(),
},
handler: async (ctx, args) => {
const identity = await ctx.auth.getUserIdentity();
if (!identity) throw new Error("Unauthorized");
// Send transactional email
return await loops.sendTransactional(ctx, {
transactionalId: "welcome-email-template-id",
email: args.email,
dataVariables: {
name: args.name,
},
});
},
});API Reference
Contact Management
Add or Update Contact
await loops.addContact(ctx, {
email: "[email protected]",
firstName: "John",
lastName: "Doe",
userId: "user123",
source: "webapp",
subscribed: true,
userGroup: "premium",
});Update Contact
await loops.updateContact(ctx, "[email protected]", {
firstName: "Jane",
userGroup: "vip",
});Find Contact
const contact = await loops.findContact(ctx, "[email protected]");List Contacts
List contacts with pagination and optional filtering.
// Simple list with default limit (100)
const result = await loops.listContacts(ctx);
// List with filters and pagination
const result = await loops.listContacts(ctx, {
userGroup: "premium",
subscribed: true,
limit: 20,
offset: 0
});
console.log(result.contacts); // Array of contacts
console.log(result.total); // Total count matching filters
console.log(result.hasMore); // Boolean indicating if more pages existDelete Contact
await loops.deleteContact(ctx, "[email protected]");Batch Create Contacts
await loops.batchCreateContacts(ctx, {
contacts: [
{ email: "[email protected]", firstName: "John" },
{ email: "[email protected]", firstName: "Jane" },
],
});Unsubscribe/Resubscribe
await loops.unsubscribeContact(ctx, "[email protected]");
await loops.resubscribeContact(ctx, "[email protected]");Count Contacts
// Count all contacts
const total = await loops.countContacts(ctx, {});
// Count by filter
const premium = await loops.countContacts(ctx, {
userGroup: "premium",
subscribed: true,
});Email Sending
Send Transactional Email
await loops.sendTransactional(ctx, {
transactionalId: "template-id-from-loops",
email: "[email protected]",
dataVariables: {
name: "John",
orderId: "12345",
},
});Send Event (Triggers Workflows)
await loops.sendEvent(ctx, {
email: "[email protected]",
eventName: "purchase_completed",
eventProperties: {
product: "Premium Plan",
amount: 99.99,
},
});Trigger Loop (Automated Sequence)
await loops.triggerLoop(ctx, {
loopId: "loop-id-from-loops",
email: "[email protected]",
dataVariables: {
onboardingStep: "welcome",
},
});Monitoring & Analytics
Get Email Statistics
const stats = await loops.getEmailStats(ctx, {
timeWindowMs: 3600000, // Last hour
});
console.log(stats.totalOperations); // Total emails sent
console.log(stats.successfulOperations); // Successful sends
console.log(stats.failedOperations); // Failed sends
console.log(stats.operationsByType); // Breakdown by type
console.log(stats.uniqueRecipients); // Unique email addressesDetect Spam Patterns
// Detect recipients with suspicious activity
const spamRecipients = await loops.detectRecipientSpam(ctx, {
timeWindowMs: 3600000,
maxEmailsPerRecipient: 10,
});
// Detect actors with suspicious activity
const spamActors = await loops.detectActorSpam(ctx, {
timeWindowMs: 3600000,
maxEmailsPerActor: 50,
});
// Detect rapid-fire patterns
const rapidFire = await loops.detectRapidFirePatterns(ctx, {
timeWindowMs: 60000, // Last minute
maxEmailsPerWindow: 5,
});Rate Limiting
Check Rate Limits
// Check recipient rate limit
const recipientCheck = await loops.checkRecipientRateLimit(ctx, {
email: "[email protected]",
timeWindowMs: 3600000, // 1 hour
maxEmails: 10,
});
if (!recipientCheck.allowed) {
throw new Error(`Rate limit exceeded. Try again after ${recipientCheck.retryAfter}ms`);
}
// Check actor rate limit
const actorCheck = await loops.checkActorRateLimit(ctx, {
actorId: "user123",
timeWindowMs: 60000, // 1 minute
maxEmails: 20,
});
// Check global rate limit
const globalCheck = await loops.checkGlobalRateLimit(ctx, {
timeWindowMs: 60000,
maxEmails: 1000,
});Example: Rate-limited email sending
export const sendTransactionalWithRateLimit = action({
args: {
transactionalId: v.string(),
email: v.string(),
actorId: v.optional(v.string()),
},
handler: async (ctx, args) => {
const identity = await ctx.auth.getUserIdentity();
if (!identity) throw new Error("Unauthorized");
const actorId = args.actorId ?? identity.subject;
// Check rate limit before sending
const rateLimitCheck = await loops.checkActorRateLimit(ctx, {
actorId,
timeWindowMs: 60000, // 1 minute
maxEmails: 10,
});
if (!rateLimitCheck.allowed) {
throw new Error(
`Rate limit exceeded. Please try again after ${rateLimitCheck.retryAfter}ms.`
);
}
// Send email
return await loops.sendTransactional(ctx, {
...args,
actorId,
});
},
});Using the API Helper
The component also exports an api() helper for easier re-exporting:
import { Loops } from "@devwithbobby/loops";
import { components } from "./_generated/api";
const loops = new Loops(components.loops);
// Export all functions at once
export const {
addContact,
updateContact,
sendTransactional,
sendEvent,
triggerLoop,
countContacts,
listContacts,
// ... all other functions
} = loops.api();Security Warning: The api() helper exports functions without authentication. Always wrap these functions with auth checks in production:
export const addContact = action({
args: { email: v.string(), ... },
handler: async (ctx, args) => {
const identity = await ctx.auth.getUserIdentity();
if (!identity) throw new Error("Unauthorized");
return await loops.addContact(ctx, args);
},
});Security Best Practices
- Always add authentication - Wrap all functions with auth checks
- Use environment variables - Store API key in Convex environment variables (never hardcode)
- Implement rate limiting - Use the built-in rate limiting queries to prevent abuse
- Monitor for abuse - Use spam detection queries to identify suspicious patterns
- Sanitize errors - Don't expose sensitive error details to clients
Authentication Example
All functions should be wrapped with authentication:
export const addContact = action({
args: { email: v.string(), ... },
handler: async (ctx, args) => {
// Add authentication check
const identity = await ctx.auth.getUserIdentity();
if (!identity) throw new Error("Unauthorized");
// Add authorization checks if needed
// if (!isAdmin(identity)) throw new Error("Forbidden");
return await loops.addContact(ctx, args);
},
});Environment Variables
Set LOOPS_API_KEY in your Convex environment:
Via CLI:
npx convex env set LOOPS_API_KEY "your-api-key"Via Dashboard:
- Go to your Convex Dashboard
- Navigate to Settings -> Environment Variables
- Add
LOOPS_API_KEYwith your Loops.so API key value
Get your API key from Loops.so Dashboard.
Never pass the API key directly in code or via function options in production. Always use environment variables.
Monitoring & Rate Limiting
The component automatically logs all email operations to the emailOperations table for monitoring. Use the built-in queries to:
- Track email statistics - See total sends, success/failure rates, breakdowns by type
- Detect spam patterns - Identify suspicious activity by recipient or actor
- Enforce rate limits - Prevent abuse with recipient, actor, or global rate limits
- Monitor for abuse - Detect rapid-fire patterns and unusual sending behavior
All monitoring queries are available through the Loops client - see the Monitoring & Analytics section above for usage examples.
Development
Local Development
To use this component in development with live reloading:
bun run dev:backendThis starts Convex dev with --live-component-sources enabled, allowing changes to be reflected immediately.
Building
npm run buildTesting
npm testProject Structure
src/
component/ # The Convex component
convex.config.ts # Component configuration
schema.ts # Database schema
lib.ts # Component functions
validators.ts # Zod validators
tables/ # Table definitions
client/ # Client library
index.ts # Loops client class
types.ts # TypeScript types
example/ # Example app
convex/
example.ts # Example usageAPI Coverage
This component implements the following Loops.so API endpoints:
- Create/Update Contact
- Delete Contact
- Find Contact
- Batch Create Contacts
- Unsubscribe/Resubscribe Contact
- Count Contacts (custom implementation)
- List Contacts (custom implementation)
- Send Transactional Email
- Send Event
- Trigger Loop
Contributing
Contributions are welcome! Please open an issue or submit a pull request.
License
Apache-2.0
