Claude-code-plugins finta-rate-limits

install
source · Clone the upstream repo
git clone https://github.com/jeremylongshore/claude-code-plugins-plus-skills
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/jeremylongshore/claude-code-plugins-plus-skills "$T" && mkdir -p ~/.claude/skills && cp -r "$T/plugins/saas-packs/finta-pack/skills/finta-rate-limits" ~/.claude/skills/jeremylongshore-claude-code-plugins-finta-rate-limits && rm -rf "$T"
manifest: plugins/saas-packs/finta-pack/skills/finta-rate-limits/SKILL.md
source content

Finta Rate Limits

Overview

Finta operates as a web-first fundraising platform without a traditional REST API, so rate limits manifest as plan-tier usage caps rather than HTTP request quotas. When building integrations through Finta's webhook events or Zapier connectors, the bottleneck is typically the number of investor pipeline records, deal room operations, and Aurora AI suggestion calls you can make within your plan tier. Teams running active fundraising rounds need to understand these ceilings to avoid mid-raise disruptions.

Rate Limit Reference

Endpoint / FeatureLimitWindowScope
Investor pipeline records50 (Free) / Unlimited (Pro)RollingPer workspace
Deal room creation1 (Free) / Unlimited (Pro)RollingPer workspace
Aurora AI suggestions10/day (Free) / 100/day (Pro)24 hoursPer user
Webhook event delivery100 events1 minutePer workspace
Data room file uploads25 (Free) / Unlimited (Pro)RollingPer deal room

Rate Limiter Implementation

class FintaUsageTracker {
  private counts: Map<string, { used: number; limit: number; resetAt: number }> = new Map();

  register(resource: string, limit: number, windowMs: number) {
    this.counts.set(resource, { used: 0, limit, resetAt: Date.now() + windowMs });
  }

  async acquire(resource: string): Promise<boolean> {
    const entry = this.counts.get(resource);
    if (!entry) throw new Error(`Unknown resource: ${resource}`);
    if (Date.now() > entry.resetAt) { entry.used = 0; entry.resetAt = Date.now() + 86_400_000; }
    if (entry.used >= entry.limit) return false;
    entry.used++;
    return true;
  }

  remaining(resource: string): number {
    const entry = this.counts.get(resource);
    return entry ? entry.limit - entry.used : 0;
  }
}

const tracker = new FintaUsageTracker();
tracker.register("aurora_suggestions", 100, 86_400_000);
tracker.register("webhook_events", 100, 60_000);

Retry Strategy

async function fintaWebhookRetry(payload: any, webhookUrl: string, maxRetries = 3): Promise<void> {
  for (let attempt = 0; attempt <= maxRetries; attempt++) {
    try {
      const res = await fetch(webhookUrl, {
        method: "POST",
        headers: { "Content-Type": "application/json" },
        body: JSON.stringify(payload),
      });
      if (res.ok) return;
      if (res.status === 429) {
        const delay = Math.pow(2, attempt) * 5000 + Math.random() * 2000;
        await new Promise(r => setTimeout(r, delay));
        continue;
      }
      throw new Error(`Webhook delivery failed: ${res.status}`);
    } catch (err) {
      if (attempt === maxRetries) throw err;
      await new Promise(r => setTimeout(r, Math.pow(2, attempt) * 3000));
    }
  }
}

Batch Processing

async function batchSyncInvestors(investors: any[], batchSize = 20) {
  const results: any[] = [];
  for (let i = 0; i < investors.length; i += batchSize) {
    const batch = investors.slice(i, i + batchSize);
    for (const investor of batch) {
      const allowed = await tracker.acquire("webhook_events");
      if (!allowed) { await new Promise(r => setTimeout(r, 60_000)); }
      results.push(await fintaWebhookRetry(investor, WEBHOOK_URL));
    }
    if (i + batchSize < investors.length) await new Promise(r => setTimeout(r, 3000));
  }
  return results;
}

Error Handling

IssueCauseFix
Plan limit reachedFree tier investor cap (50)Upgrade to Pro or archive inactive investors
Aurora AI unavailableDaily suggestion quota exhaustedWait for 24h reset or upgrade plan
Webhook delivery 429Burst of pipeline eventsQueue events, deliver with 1s spacing
Deal room lockedFree plan single-room limitClose existing room before opening new one
File upload rejectedData room storage cap on Free tierCompress files or upgrade plan

Resources

Next Steps

See

finta-performance-tuning
.