Auto-commit 2026-04-27 12:34

This commit is contained in:
2026-04-27 12:34:30 -04:00
parent bef1d7f829
commit 9e3a54f508
12 changed files with 1025 additions and 5 deletions

View File

@@ -162,3 +162,41 @@
---
**Status:** FRE-710, FRE-670 complete. FRE-713 in progress (CTO). FRE-627 blocked on deployment.
## FRE-750: Recovery Cascade Cleanup ✅
**Status:** COMPLETED - 2026-04-27 05:20 UTC
**Problem:**
Infinite recovery cascade triggered by FRE-620 (analytics setup) when Senior Engineer went into error state:
- FRE-620 assigned to Senior Engineer (error state) → stalled
- FRE-750 created to recover FRE-620
- FRE-767 created to recover FRE-750 (recovery of recovery)
- Cascade continued: FRE-779 → FRE-789 → ... → FRE-2000+ (700+ issues)
- All assigned to CEO, who cannot perform engineering work
- Recovery system in runaway loop creating new issues faster than they could be cancelled
**Root Cause:**
Paperclip recovery invariant `stranded_assigned_issue` was creating recovery issues for:
1. Recovery issues (not just original work issues)
2. Cancelled recovery issues (system bug)
**Resolution:**
1. Broke blocker chain on FRE-750
2. Reassigned FRE-620 to Founding Engineer (available, running status)
3. Cancelled ~700 recovery issues in batch operations
4. Marked FRE-750 as done
**Final State:**
| Issue | Status | Owner |
|-------|--------|-------|
| FRE-620 (Analytics) | in_progress | Founding Engineer |
| FRE-750 (Recovery) | done | CEO |
| FRE-767+ (Cascade) | cancelled | - |
**System Bug Documented:**
Recovery system needs fix to prevent creating recovery issues for:
- Already-cancelled issues
- Recovery issues (should only recover original work)
**Git Commit:** bef1d7f8 - "FRE-750: Break infinite recovery cascade"

View File

@@ -0,0 +1,139 @@
# FRE-674 Handoff to CTO
**From:** CMO
**To:** CTO
**Date:** April 27, 2026
**Status:** ✅ Implementation Complete - Awaiting Dashboard
---
## What's Done
### UTM Tracking Implementation
**Backend:**
- ✅ Added UTM parameters to `server/trpc/beta-router.ts`
- ✅ Parameters: utmSource, utmMedium, utmCampaign, utmContent, utmTerm
- ✅ All data stored in `waitlistSignups.metadata` JSON field
**Frontend:**
- ✅ Added automatic UTM capture to `src/routes/beta/BetaSignup.tsx`
- ✅ No user action required - extracts from URL automatically
- ✅ TypeScript types updated in `src/lib/api/trpc-hooks.ts`
**Code Quality:**
- ✅ TypeScript compilation passes
- ✅ No breaking changes to existing signup flow
---
## What's Next
### CTO Dashboard Implementation (Due: April 30)
**File:** `/marketing/reddit-campaign-utm-tracking.md` (Section: Analytics Dashboard Setup)
**Tasks:**
1. **Create Reddit campaign dashboard view**
- Filter by utm_campaign = "beta_recruitment"
- Filter by utm_source = "reddit"
- Date range: May 3-9, 2026
2. **Add the following visualizations:**
- Daily applications by subreddit (bar chart)
- Conversion funnel (page view → form start → submit)
- Cumulative applications (line chart, target: 100)
3. **Set up conversion events:**
- Form start event
- Form submit event
- Track by utm_content (subreddit)
4. **Share dashboard access with CMO**
**Metrics to Display:**
| Metric | Description |
|--------|-------------|
| Page views by utm_content | Traffic per subreddit |
| Form starts by utm_content | Engagement per subreddit |
| Form completions by utm_content | Applications per subreddit |
| Conversion rate | Applications / Page views |
| Total Reddit traffic | Aggregate across all subreddits |
| Total applications | Campaign total |
| Overall conversion rate | Campaign efficiency |
---
## Testing Plan (CMO - April 28-30)
Once dashboard is ready, CMO will:
1. Test all 3 tracking URLs manually
2. Submit test applications
3. Verify data appears in dashboard
4. Validate conversion tracking works
**Test URLs:**
```
r/Screenwriting: scripter.app/beta?utm_source=reddit&utm_content=screenwriting
r/Filmmakers: scripter.app/beta?utm_source=reddit&utm_content=filmmakers
r/Scriptwriting: scripter.app/beta?utm_source=reddit&utm_content=scriptwriting
```
---
## Campaign Timeline
- **April 28-30:** Testing phase
- **May 3:** Campaign launch (r/Screenwriting + r/Filmmakers)
- **May 4:** AMA day
- **May 6:** Update post (60/100 filled)
- **May 8:** r/Scriptwriting post
- **May 9:** Campaign wrap
- **May 10-12:** Post-campaign analysis
---
## Database Verification
To verify UTM data is being captured correctly:
```sql
SELECT
email,
name,
metadata->>'$.utmSource' as utm_source,
metadata->>'$.utmCampaign' as utm_campaign,
metadata->>'$.utmContent' as utm_content,
createdAt
FROM waitlistSignups
WHERE metadata LIKE '%utmSource%'
ORDER BY createdAt DESC;
```
**Expected metadata JSON:**
```json
{
"isBetaApplication": true,
"utmSource": "reddit",
"utmMedium": "social",
"utmCampaign": "beta_recruitment",
"utmContent": "screenwriting",
"primaryRole": "...",
...
}
```
---
## Acceptance Criteria
- [ ] Dashboard displays UTM data by subreddit
- [ ] Conversion funnel tracking works
- [ ] Daily metrics visible in real-time
- [ ] CMO has access to dashboard
- [ ] Testing confirms data accuracy
---
**Next Action:** CTO implements analytics dashboard (due April 30)
**Blocker:** None - implementation ready for dashboard integration

View File

@@ -351,3 +351,44 @@ Recovered from terminal run failure (process_lost_retry). All deliverables intac
| 2026-05-03 | Post date (if approved) |
**Status:** 🟢 EXECUTED - Awaiting mod response
## FRE-673 Continued: r/Filmmakers Cross-Post - EXECUTED (April 27)
**Status:** ✅ MESSAGE SENT
**Time:** 2026-04-27 (Monday afternoon PT)
**Priority:** MEDIUM
### Action Taken
**Sent mod mail to r/Filmmakers (200K members)**
- URL: https://www.reddit.com/message/compose?to=%2Fr%2Filmmakers
- Subject: "Request: Cross-post for screenwriters in your community"
- Message: Customized cross-post request highlighting writer-director collaboration benefits
### Message Content Summary
**Key points covered:**
- Cross-post permission request (not primary post)
- Real-time collaboration feature (appeals to writer-director teams)
- Connection to r/Screenwriting primary post
- Willingness to tailor message to community guidelines
### Files Updated
- `/marketing/reddit-mod-outreach-tracker.md` - r/Filmmakers marked as SENT
### Next Steps
**Wait for mod response (24-48 hours expected):**
- April 30: Follow up if no response
- Coordinate with r/Screenwriting approval status
### Status
**Progress:** 2/3 subreddits contacted
- ✅ r/Screenwriting (PRIMARY) - Pending response
- ✅ r/Filmmakers (SECONDARY) - Pending response
- ⏳ r/Scriptwriting (TERTIARY) - Ready to send
**Status:** 🟢 EXECUTED - Awaiting mod responses

View File

@@ -12,7 +12,7 @@
| Subreddit | Members | Contacted | Response | Approved | Notes |
|-----------|---------|-----------|----------|----------|-------|
| r/Screenwriting | 500K | ✅ Contacted 4/27 | ⏳ Pending | - | PRIMARY - FRE-673 in progress |
| r/Filmmakers | 200K | ⏳ Ready to send | - | - | Cross-post permission |
| r/Filmmakers | 200K | ✅ Contacted 4/27 | ⏳ Pending | - | Cross-post permission |
| r/Scriptwriting | 30K | ⏳ Ready to send | - | - | Smaller sub, backup |
---
@@ -145,7 +145,7 @@ Thanks!
| Date/Time | Subreddit | Message Sent | Mod Response | Status |
|-----------|-----------|--------------|--------------|--------|
| 2026-04-27 | r/Screenwriting | ✅ SENT | ⏳ Pending | In Progress - FRE-673 |
| [Fill in] | r/Filmmakers | ⏳ Ready | - | Pending |
| 2026-04-27 | r/Filmmakers | ✅ SENT | ⏳ Pending | In Progress - FRE-673 |
| [Fill in] | r/Scriptwriting | ⏳ Ready | - | Pending |
---

View File

@@ -33,14 +33,14 @@
## FRE-674: Reddit Campaign UTM Tracking
**Status:** IMPLEMENTATION COMPLETE
**Status:** ✅ COMPLETE
**Work Done:**
- Added UTM parameter capture to beta signup form
- Backend: Updated `/server/trpc/beta-router.ts` to accept utmSource, utmMedium, utmCampaign, utmContent, utmTerm
- Frontend: Updated `/src/routes/beta/BetaSignup.tsx` to extract UTM params from URL automatically
- Updated hook types in `/src/lib/api/trpc-hooks.ts`
- All UTM data stored in waitlistSignups.metadata JSON field
- TypeScript compilation passes with no errors
**Testing Plan:**
- Manual testing scheduled April 28-30
@@ -49,8 +49,9 @@
**Documentation:**
- Updated `/marketing/reddit-campaign-utm-tracking.md` with implementation details and testing guide
- Created `/agents/cmo/FRE-674-cto-handoff.md` for CTO dashboard implementation
**Next Actions:**
**Handoff:**
- CTO: Implement analytics dashboard to visualize UTM data (due April 30)
- CMO: Test tracking URLs April 28-30
- CMO: Monitor Reddit campaign performance May 3-9

View File

@@ -0,0 +1,258 @@
# Analytics Event Taxonomy
## Overview
This document defines the event taxonomy for FrenoCorp's analytics implementation across Mixpanel, GA4, and Stripe. Events are organized by category and mapped to each platform.
## Event Categories
### 1. User Events
| Event Name | Mixpanel | GA4 | Stripe | Description |
|------------|----------|-----|--------|-------------|
| `user_signedup` | ✅ | `sign_up` | `customer.created` | New user registration |
| `user_signedin` | ✅ | `login` | - | User login |
| `user_signedout` | ✅ | `logout` | - | User logout |
| `user_profile_updated` | ✅ | `update_user` | `customer.updated` | Profile changes |
**Properties:**
- `user_id` (string)
- `email` (string)
- `signup_method` (string: 'email', 'google', 'github')
- `plan_tier` (string: 'free', 'pro', 'enterprise')
- `timestamp` (ISO 8601)
### 2. Project Events
| Event Name | Mixpanel | GA4 | Stripe | Description |
|------------|----------|-----|--------|-------------|
| `project_created` | ✅ | `project_create` | - | New project created |
| `project_updated` | ✅ | `project_update` | - | Project modifications |
| `project_deleted` | ✅ | `project_delete` | - | Project removal |
| `project_shared` | ✅ | `share_project` | - | Sharing with collaborators |
**Properties:**
- `project_id` (string)
- `project_name` (string)
- `template_id` (string, optional)
- `collaborator_count` (number)
- `timestamp` (ISO 8601)
### 3. Screenplay Events
| Event Name | Mixpanel | GA4 | Stripe | Description |
|------------|----------|-----|--------|-------------|
| `screenplay_created` | ✅ | `screenplay_create` | - | New screenplay |
| `screenplay_updated` | ✅ | `screenplay_update` | - | Screenplay modification |
| `screenplay_exported` | ✅ | `screenplay_export` | - | Export to PDF/Word |
| `screenplay_saved` | ✅ | `save_screenplay` | - | Auto-save event |
**Properties:**
- `screenplay_id` (string)
- `project_id` (string)
- `page_count` (number)
- `word_count` (number)
- `export_format` (string, optional: 'pdf', 'docx', 'fdx')
- `timestamp` (ISO 8601)
### 4. Collaboration Events
| Event Name | Mixpanel | GA4 | Stripe | Description |
|------------|----------|-----|--------|-------------|
| `invitation_sent` | ✅ | `send_invitation` | - | Invite sent to collaborator |
| `invitation_accepted` | ✅ | `accept_invitation` | - | Collaborator joined |
| `collaboration_started` | ✅ | `start_collaboration` | - | Real-time editing began |
| `collaboration_updated` | ✅ | `update_collaboration` | - | Collaboration settings changed |
**Properties:**
- `project_id` (string)
- `inviter_id` (string)
- `invitee_email` (string)
- `permission_level` (string: 'view', 'edit', 'admin')
- `timestamp` (ISO 8601)
### 5. Export/Import Events
| Event Name | Mixpanel | GA4 | Stripe | Description |
|------------|----------|-----|--------|-------------|
| `export_completed` | ✅ | `export_complete` | - | Export finished |
| `export_failed` | ✅ | `export_error` | - | Export error |
| `import_completed` | ✅ | `import_complete` | - | Import finished |
| `import_failed` | ✅ | `import_error` | - | Import error |
**Properties:**
- `export_format` (string: 'pdf', 'docx', 'fdx', 'txt')
- `file_size_bytes` (number)
- `duration_ms` (number)
- `success` (boolean)
- `error_message` (string, optional)
- `timestamp` (ISO 8601)
### 6. Subscription/Payment Events
| Event Name | Mixpanel | GA4 | Stripe | Description |
|------------|----------|-----|--------|-------------|
| `subscription_started` | ✅ | `begin_checkout` | `subscription.created` | Subscription initiated |
| `subscription_renewed` | ✅ | `purchase` | `invoice.payment_succeeded` | Subscription renewed |
| `subscription_cancelled` | ✅ | `purchase` | `subscription.deleted` | Subscription cancelled |
| `payment_succeeded` | ✅ | `purchase` | `payment_intent.succeeded` | Payment successful |
| `payment_failed` | ✅ | `purchase` | `payment_intent.payment_failed` | Payment failed |
| `upgrade_attempted` | ✅ | `select_item` | `customer.updated` | Plan upgrade |
**Properties:**
- `subscription_id` (string)
- `customer_id` (string)
- `plan_tier` (string: 'free', 'pro', 'enterprise')
- `amount` (number, in cents)
- `currency` (string: 'USD', 'EUR', etc.)
- `payment_method` (string: 'card', 'paypal', 'bank_transfer')
- `duration_months` (number)
- `timestamp` (ISO 8601)
### 7. E-commerce Events (GA4 Enhanced)
| Event Name | Mixpanel | GA4 | Stripe | Description |
|------------|----------|-----|--------|-------------|
| `view_pricing_page` | ✅ | `view_item` | - | Viewed pricing |
| `add_to_cart` | ✅ | `add_to_cart` | - | Selected plan |
| `begin_checkout` | ✅ | `begin_checkout` | - | Started checkout |
| `purchase` | ✅ | `purchase` | `invoice.payment_succeeded` | Completed purchase |
**E-commerce Item Properties:**
```typescript
interface GA4EcommerceItem {
item_id: string; // Plan SKU (e.g., 'pro_monthly')
item_name: string; // Display name (e.g., 'Pro Monthly')
item_category: string; // 'subscription_tier'
price: number; // Price in cents
quantity: number; // Usually 1 for subscriptions
currency: string; // 'USD', 'EUR', etc.
}
```
### 8. Engagement Events
| Event Name | Mixpanel | GA4 | Stripe | Description |
|------------|----------|-----|--------|-------------|
| `page_view` | ✅ | `page_view` | - | Page navigation |
| `search` | ✅ | `search` | - | Search performed |
| `feature_used` | ✅ | `feature_click` | - | Feature interaction |
| `help_viewed` | ✅ | `view_help` | - | Help documentation |
**Properties:**
- `page_name` (string)
- `search_query` (string, optional)
- `feature_name` (string, optional)
- `session_id` (string)
- `timestamp` (ISO 8601)
## Platform Configuration
### Mixpanel
```typescript
// Project Token: Environment variable MIXPANEL_PROJECT_TOKEN
// Options:
{
debug: process.env.NODE_ENV === 'development',
track_pageview: true,
persistence: 'localStorage',
}
```
**Key Features:**
- User identification with `identify()`
- User properties with `people.set()`
- Group analytics with `group()`
- Cohort analysis support
### GA4
```typescript
// Measurement ID: Environment variable GA4_MEASUREMENT_ID
// Options:
{
debug: process.env.NODE_ENV === 'development',
autoTrackPageViews: true,
autoTrackScrolls: true,
autoTrackOutboundLinks: true,
}
```
**Key Features:**
- Enhanced e-commerce tracking
- Automatic pageview tracking
- Scroll depth tracking
- Outbound link tracking
### Stripe
```typescript
// Configuration:
{
publishableKey: process.env.STRIPE_PUBLISHABLE_KEY,
secretKey: process.env.STRIPE_SECRET_KEY,
webhookSecret: process.env.STRIPE_WEBHOOK_SECRET,
apiVersion: '2024-12-18.acacia',
}
```
**Key Features:**
- Customer management
- Subscription lifecycle
- Payment intent tracking
- Webhook event handling
## Implementation Notes
### Event Naming Conventions
1. **Mixpanel**: snake_case (e.g., `user_signedup`)
2. **GA4**: snake_case for custom events, standard names for e-commerce
3. **Stripe**: dot notation (e.g., `customer.created`)
### Property Naming
- Use snake_case for all property names
- Use ISO 8601 for timestamps
- Use lowercase for enum values
- Include `timestamp` on all events
### Data Retention
- **Mixpanel**: 24 months (default), configurable
- **GA4**: 14 months (default), configurable
- **Stripe**: Indefinite for subscription data
### Privacy Considerations
- PII stored in Mixpanel: `email`, `name`
- PII stored in GA4: User properties only (not in events)
- PII stored in Stripe: Full customer data
## Environment Variables
```bash
# Mixpanel
MIXPANEL_PROJECT_TOKEN=
# GA4
GA4_MEASUREMENT_ID=
# Stripe
STRIPE_PUBLISHABLE_KEY=
STRIPE_SECRET_KEY=
STRIPE_WEBHOOK_SECRET=
```
## Version History
| Version | Date | Changes |
|---------|------|---------|
| 1.0 | 2026-04-27 | Initial taxonomy for FRE-620 |
## Related Documents
- Parent Issue: [FRE-585](/FRE/issues/FRE-585) - Analytics dashboard setup and KPI tracking
- Implementation: `src/lib/analytics/`
- KPI Definitions: `src/lib/analytics/kpi-service.ts`

View File

@@ -0,0 +1,93 @@
import { createMixpanelService, type MixpanelConfig } from "./mixpanel-service";
import { createGA4Service, type GA4Config } from "./ga4-service";
import { createStripeService, type StripeDashboardConfig } from "./stripe-service";
export interface AnalyticsConfig {
mixpanel: MixpanelConfig;
ga4: GA4Config;
stripe: StripeDashboardConfig;
}
export class AnalyticsConfigManager {
private config: AnalyticsConfig;
private mixpanelService?: ReturnType<typeof createMixpanelService>;
private ga4Service?: ReturnType<typeof createGA4Service>;
private stripeService?: ReturnType<typeof createStripeService>;
constructor(config: AnalyticsConfig) {
this.config = config;
}
initialize(): void {
// Initialize Mixpanel
this.mixpanelService = createMixpanelService(this.config.mixpanel);
// Initialize GA4
this.ga4Service = createGA4Service(this.config.ga4);
this.ga4Service.initialize();
// Initialize Stripe
this.stripeService = createStripeService(this.config.stripe);
this.stripeService.initialize();
}
getMixpanel(): ReturnType<typeof createMixpanelService> {
if (!this.mixpanelService) {
throw new Error("Analytics not initialized. Call initialize() first.");
}
return this.mixpanelService;
}
getGA4(): ReturnType<typeof createGA4Service> {
if (!this.ga4Service) {
throw new Error("Analytics not initialized. Call initialize() first.");
}
return this.ga4Service;
}
getStripe(): ReturnType<typeof createStripeService> {
if (!this.stripeService) {
throw new Error("Analytics not initialized. Call initialize() first.");
}
return this.stripeService;
}
getConfig(): AnalyticsConfig {
return this.config;
}
}
export const loadAnalyticsConfig = (): AnalyticsConfig => {
return {
mixpanel: {
projectToken: import.meta.env.VITE_MIXPANEL_PROJECT_TOKEN || "",
options: {
debug: import.meta.env.VITE_MIXPANEL_DEBUG === "true",
track_pageview: true,
persistence: "localStorage",
},
},
ga4: {
measurementId: import.meta.env.VITE_GA4_MEASUREMENT_ID || "",
options: {
debug: import.meta.env.VITE_GA4_DEBUG === "true",
autoTrackPageViews: true,
autoTrackScrolls: true,
autoTrackOutboundLinks: true,
},
},
stripe: {
publishableKey: import.meta.env.VITE_STRIPE_PUBLISHABLE_KEY || "",
secretKey: import.meta.env.VITE_STRIPE_SECRET_KEY || "",
webhookSecret: import.meta.env.VITE_STRIPE_WEBHOOK_SECRET || "",
apiVersion: "2024-12-18.acacia",
},
};
};
export const createAnalyticsManager = (): AnalyticsConfigManager => {
const config = loadAnalyticsConfig();
const manager = new AnalyticsConfigManager(config);
manager.initialize();
return manager;
};

View File

@@ -0,0 +1,78 @@
export interface GA4LoadOptions {
debug?: boolean;
autoTrackPageViews?: boolean;
autoTrackScrolls?: boolean;
autoTrackOutboundLinks?: boolean;
}
export const loadGA4 = async (
measurementId: string,
options: GA4LoadOptions = {}
): Promise<void> => {
const {
debug = false,
autoTrackPageViews = true,
autoTrackScrolls = true,
autoTrackOutboundLinks = false,
} = options;
// Load GA4 script
await new Promise<void>((resolve) => {
const script = document.createElement("script");
script.async = true;
script.src = `https://www.googletagmanager.com/gtag/js?id=${measurementId}`;
script.onload = () => resolve();
document.head.appendChild(script);
});
// Initialize gtag
window.dataLayer = window.dataLayer || [];
(window as any).gtag = function (...args: unknown[]) {
(window as any).dataLayer.push(args);
};
// Configure GA4
(window as any).gtag("config", measurementId, {
debug_mode: debug,
send_page_view: autoTrackPageViews,
});
// Auto-track scrolls
if (autoTrackScrolls) {
const scrollDepth = [25, 50, 75, 100];
scrollDepth.forEach((threshold) => {
const observer = new IntersectionObserver(
(entries) => {
entries.forEach((entry) => {
if (entry.isIntersecting) {
(window as any).gtag("event", "scroll", {
scroll_percentage: threshold,
});
observer.unobserve(entry.target);
}
});
},
{ threshold: threshold / 100 }
);
observer.observe(document.body);
});
}
// Auto-track outbound links
if (autoTrackOutboundLinks) {
document.addEventListener("click", (e) => {
const target = e.target as HTMLAnchorElement;
if (target.tagName === "A" && target.href) {
const currentHost = window.location.hostname;
const linkHost = new URL(target.href).hostname;
if (linkHost !== currentHost) {
(window as any).gtag("event", "click", {
event_category: "outbound_link",
event_label: target.href,
value: 1,
});
}
}
});
}
};

View File

@@ -0,0 +1,139 @@
import { loadGA4 } from "./ga4-loader";
type GA4Event =
| "session_start"
| "page_view"
| "scroll"
| "click"
| "user_signup"
| "user_login"
| "user_logout"
| "project_create"
| "project_update"
| "project_delete"
| "screenplay_create"
| "screenplay_update"
| "screenplay_export"
| "begin_checkout"
| "add_to_cart"
| "add_to_wishlist"
| "view_item"
| "view_item_list"
| "select_item"
| "purchase"
| "refund"
| "sign_up"
| "lead"
| "search";
export interface GA4Config {
measurementId: string;
apiSecret?: string;
streamId?: string;
options?: {
debug?: boolean;
autoTrackPageViews?: boolean;
autoTrackScrolls?: boolean;
autoTrackOutboundLinks?: boolean;
};
}
export interface GA4EcommerceItem {
item_id: string;
item_name: string;
item_category?: string;
price: number;
quantity: number;
currency?: string;
}
export interface GA4PurchaseEvent {
transaction_id: string;
value: number;
currency: string;
tax?: number;
shipping?: number;
items: GA4EcommerceItem[];
}
export class GA4Service {
private measurementId: string;
private config: GA4Config;
private initialized: boolean = false;
constructor(config: GA4Config) {
this.measurementId = config.measurementId;
this.config = config;
}
async initialize(): Promise<void> {
await loadGA4(this.measurementId, this.config.options);
this.initialized = true;
}
track(event: GA4Event, params?: Record<string, unknown>): void {
if (!this.initialized) {
console.warn("GA4 not initialized");
return;
}
(window as any).gtag?.("event", event, params);
}
trackPageView(pageLocation?: string, pagePath?: string): void {
this.track("page_view", {
page_location: pageLocation || window.location.href,
page_path: pagePath || window.location.pathname,
page_title: document.title,
});
}
trackEcommerce(event: "purchase" | "refund", data: GA4PurchaseEvent): void {
this.track(event, {
transaction_id: data.transaction_id,
value: data.value,
currency: data.currency,
tax: data.tax,
shipping: data.shipping,
items: data.items,
});
}
trackAddToCart(item: GA4EcommerceItem): void {
this.track("add_to_cart", {
items: [item],
});
}
trackBeginCheckout(items: GA4EcommerceItem[], value: number): void {
this.track("begin_checkout", {
items,
value,
currency: items[0]?.currency || "USD",
});
}
trackViewItemList(items: GA4EcommerceItem[], itemListName: string): void {
this.track("view_item_list", {
items,
item_list_name: itemListName,
});
}
trackSelectItem(item: GA4EcommerceItem, itemListName: string): void {
this.track("select_item", {
items: [item],
item_list_name: itemListName,
});
}
setUserProperty(userId: string, userProperties?: Record<string, string>): void {
(window as any).gtag?.("set", "user_properties", {
user_id: userId,
...userProperties,
});
}
}
export const createGA4Service = (config: GA4Config): GA4Service => {
return new GA4Service(config);
};

View File

@@ -3,3 +3,7 @@ export * from "./slack-alerts";
export * from "./report-generator";
export * from "./cohort-analysis";
export * from "./nps-service";
export * from "./mixpanel-service";
export * from "./ga4-service";
export * from "./ga4-loader";
export * from "./stripe-service";

View File

@@ -0,0 +1,84 @@
import mixpanel, { type MixpanelInstance } from "mixpanel-browser";
type MixpanelEvent =
| "user_signedup"
| "user_signedin"
| "user_signedout"
| "project_created"
| "project_updated"
| "project_deleted"
| "screenplay_created"
| "screenplay_updated"
| "screenplay exported"
| "subscription_started"
| "subscription_renewed"
| "subscription_cancelled"
| "payment_succeeded"
| "payment_failed"
| "invitation_sent"
| "collaboration_started"
| "collaboration_updated"
| "export_completed"
| "import_completed";
export interface MixpanelConfig {
projectToken: string;
options?: {
debug?: boolean;
track_pageview?: boolean;
persistence?: "localStorage" | "cookie" | "memory";
loaded?: (mixpanel: MixpanelInstance) => void;
};
}
export class MixpanelService {
private mixpanel: MixpanelInstance;
private initialized: boolean = false;
constructor(config: MixpanelConfig) {
this.mixpanel = mixpanel.init(config.projectToken, config.options);
this.initialized = true;
}
track(event: MixpanelEvent, properties?: Record<string, unknown>): void {
if (!this.initialized) {
console.warn("Mixpanel not initialized");
return;
}
this.mixpanel.track(event, properties);
}
identify(userId: string, properties?: Record<string, unknown>): void {
this.mixpanel.identify(userId);
if (properties) {
this.mixpanel.people.set(userId, properties);
}
}
alias(distinctId: string, alias: string): void {
this.mixpanel.alias(alias, distinctId);
}
reset(): void {
this.mixpanel.reset();
}
getDistinctId(): string {
return this.mixpanel.get_distinct_id();
}
group(groupType: string, groupId: string): void {
this.mixpanel.group(groupType, groupId);
}
trackWithTimestamp(event: MixpanelEvent, timestamp: Date, properties?: Record<string, unknown>): void {
this.mixpanel.track(event, {
...properties,
timestamp: timestamp.toISOString(),
});
}
}
export const createMixpanelService = (config: MixpanelConfig): MixpanelService => {
return new MixpanelService(config);
};

View File

@@ -0,0 +1,145 @@
import Stripe from "stripe";
export interface StripeWebhookConfig {
secret: string;
apiVersion?: string;
}
export interface StripeDashboardConfig {
publishableKey: string;
secretKey: string;
webhookSecret: string;
apiVersion?: string;
}
type StripeEvent =
| "customer.created"
| "customer.updated"
| "customer.deleted"
| "subscription.created"
| "subscription.updated"
| "subscription.deleted"
| "invoice.created"
| "invoice.payment_succeeded"
| "invoice.payment_failed"
| "payment_intent.created"
| "payment_intent.succeeded"
| "payment_intent.payment_failed"
| "checkout.session.completed"
| "product.created"
| "product.updated"
| "price.created"
| "price.updated";
export class StripeService {
private stripe: Stripe;
private webhookSecret: string;
private initialized: boolean = false;
constructor(config: StripeDashboardConfig) {
this.stripe = new Stripe(config.secretKey, {
apiVersion: config.apiVersion || "2024-12-18.acacia",
});
this.webhookSecret = config.webhookSecret;
this.initialized = true;
}
async initialize(): Promise<void> {
// Verify connection by fetching account
const account = await this.stripe.account.retrieve();
this.initialized = !!account.id;
}
async createCustomer(email: string, name?: string, metadata?: Record<string, string>): Promise<Stripe.Customer> {
return await this.stripe.customers.create({
email,
name,
metadata,
});
}
async createSubscription(customerId: string, priceId: string): Promise<Stripe.Subscription> {
return await this.stripe.subscriptions.create({
customer: customerId,
items: [{ price: priceId }],
});
}
async createCheckoutSession(
customerId: string,
priceId: string,
successUrl: string,
cancelUrl: string
): Promise<Stripe.Checkout.Session> {
return await this.stripe.checkout.sessions.create({
customer: customerId,
line_items: [{ price: priceId, quantity: 1 }],
mode: "subscription",
success_url: successUrl,
cancel_url: cancelUrl,
});
}
async verifyWebhook(payload: Buffer, signature: string): Promise<Stripe.Event> {
return await this.stripe.webhooks.constructEventAsync(payload, signature, this.webhookSecret);
}
handleWebhookEvent(event: Stripe.Event): void {
switch (event.type) {
case "customer.created":
console.log("Customer created:", (event.data.object as Stripe.Customer).email);
break;
case "subscription.created":
console.log("Subscription created:", (event.data.object as Stripe.Subscription).id);
break;
case "invoice.payment_succeeded":
console.log("Payment succeeded:", (event.data.object as Stripe.Invoice).id);
break;
case "invoice.payment_failed":
console.log("Payment failed:", (event.data.object as Stripe.Invoice).id);
break;
case "payment_intent.succeeded":
console.log("Payment intent succeeded:", (event.data.object as Stripe.PaymentIntent).id);
break;
case "payment_intent.payment_failed":
console.log("Payment intent failed:", (event.data.object as Stripe.PaymentIntent).id);
break;
case "checkout.session.completed":
console.log("Checkout session completed:", (event.data.object as Stripe.Checkout.Session).id);
break;
default:
console.log(`Unhandled event type: ${event.type}`);
}
}
async listCustomers(limit?: number): Promise<Stripe.ApiList<Stripe.Customer>> {
return await this.stripe.customers.list({ limit: limit || 10 });
}
async getCustomer(customerId: string): Promise<Stripe.Customer> {
return await this.stripe.customers.retrieve(customerId);
}
async updateCustomer(
customerId: string,
updates: { email?: string; name?: string; metadata?: Record<string, string> }
): Promise<Stripe.Customer> {
return await this.stripe.customers.update(customerId, updates);
}
async deleteCustomer(customerId: string): Promise<Stripe.Customer> {
return await this.stripe.customers.del(customerId);
}
getDashboardUrl(): string {
return "https://dashboard.stripe.com";
}
getWebhookEndpointUrl(): string {
return `/api/webhooks/stripe`;
}
}
export const createStripeService = (config: StripeDashboardConfig): StripeService => {
return new StripeService(config);
};