lemonsqueezy-mcp-server 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (97) hide show
  1. package/ENV_EXAMPLE.md +83 -0
  2. package/LICENSE +21 -0
  3. package/README.md +141 -0
  4. package/SETUP.md +244 -0
  5. package/dist/config.d.ts +25 -0
  6. package/dist/config.d.ts.map +1 -0
  7. package/dist/config.js +80 -0
  8. package/dist/config.js.map +1 -0
  9. package/dist/config.test.d.ts +2 -0
  10. package/dist/config.test.d.ts.map +1 -0
  11. package/dist/config.test.js +55 -0
  12. package/dist/config.test.js.map +1 -0
  13. package/dist/connections/firebase.d.ts +3 -0
  14. package/dist/connections/firebase.d.ts.map +1 -0
  15. package/dist/connections/firebase.js +62 -0
  16. package/dist/connections/firebase.js.map +1 -0
  17. package/dist/connections/salesforce.d.ts +3 -0
  18. package/dist/connections/salesforce.d.ts.map +1 -0
  19. package/dist/connections/salesforce.js +307 -0
  20. package/dist/connections/salesforce.js.map +1 -0
  21. package/dist/index.d.ts +3 -0
  22. package/dist/index.d.ts.map +1 -0
  23. package/dist/index.js +101 -0
  24. package/dist/index.js.map +1 -0
  25. package/dist/resources/payment-context.d.ts +14 -0
  26. package/dist/resources/payment-context.d.ts.map +1 -0
  27. package/dist/resources/payment-context.js +111 -0
  28. package/dist/resources/payment-context.js.map +1 -0
  29. package/dist/server.d.ts +36 -0
  30. package/dist/server.d.ts.map +1 -0
  31. package/dist/server.js +14 -0
  32. package/dist/server.js.map +1 -0
  33. package/dist/sync-last-purchase.d.ts +3 -0
  34. package/dist/sync-last-purchase.d.ts.map +1 -0
  35. package/dist/sync-last-purchase.js +113 -0
  36. package/dist/sync-last-purchase.js.map +1 -0
  37. package/dist/tools/definitions.d.ts +3768 -0
  38. package/dist/tools/definitions.d.ts.map +1 -0
  39. package/dist/tools/definitions.js +753 -0
  40. package/dist/tools/definitions.js.map +1 -0
  41. package/dist/tools/handlers/analytics.d.ts +21 -0
  42. package/dist/tools/handlers/analytics.d.ts.map +1 -0
  43. package/dist/tools/handlers/analytics.js +398 -0
  44. package/dist/tools/handlers/analytics.js.map +1 -0
  45. package/dist/tools/handlers/lemonsqueezy.d.ts +436 -0
  46. package/dist/tools/handlers/lemonsqueezy.d.ts.map +1 -0
  47. package/dist/tools/handlers/lemonsqueezy.js +481 -0
  48. package/dist/tools/handlers/lemonsqueezy.js.map +1 -0
  49. package/dist/tools/handlers/salesforce.d.ts +13 -0
  50. package/dist/tools/handlers/salesforce.d.ts.map +1 -0
  51. package/dist/tools/handlers/salesforce.js +54 -0
  52. package/dist/tools/handlers/salesforce.js.map +1 -0
  53. package/dist/tools/handlers/vos.d.ts +26 -0
  54. package/dist/tools/handlers/vos.d.ts.map +1 -0
  55. package/dist/tools/handlers/vos.js +207 -0
  56. package/dist/tools/handlers/vos.js.map +1 -0
  57. package/dist/tools/index.d.ts +7 -0
  58. package/dist/tools/index.d.ts.map +1 -0
  59. package/dist/tools/index.js +202 -0
  60. package/dist/tools/index.js.map +1 -0
  61. package/dist/types.d.ts +24 -0
  62. package/dist/types.d.ts.map +1 -0
  63. package/dist/types.js +2 -0
  64. package/dist/types.js.map +1 -0
  65. package/dist/utils/logger.d.ts +3 -0
  66. package/dist/utils/logger.d.ts.map +1 -0
  67. package/dist/utils/logger.js +8 -0
  68. package/dist/utils/logger.js.map +1 -0
  69. package/dist/utils/response.d.ts +14 -0
  70. package/dist/utils/response.d.ts.map +1 -0
  71. package/dist/utils/response.js +17 -0
  72. package/dist/utils/response.js.map +1 -0
  73. package/dist/utils/retry.d.ts +11 -0
  74. package/dist/utils/retry.d.ts.map +1 -0
  75. package/dist/utils/retry.js +36 -0
  76. package/dist/utils/retry.js.map +1 -0
  77. package/dist/utils/secrets/provider.d.ts +30 -0
  78. package/dist/utils/secrets/provider.d.ts.map +1 -0
  79. package/dist/utils/secrets/provider.js +51 -0
  80. package/dist/utils/secrets/provider.js.map +1 -0
  81. package/dist/utils/secrets.d.ts +7 -0
  82. package/dist/utils/secrets.d.ts.map +1 -0
  83. package/dist/utils/secrets.js +16 -0
  84. package/dist/utils/secrets.js.map +1 -0
  85. package/dist/utils/validation.d.ts +580 -0
  86. package/dist/utils/validation.d.ts.map +1 -0
  87. package/dist/utils/validation.js +250 -0
  88. package/dist/utils/validation.js.map +1 -0
  89. package/dist/webhooks/listener.d.ts +17 -0
  90. package/dist/webhooks/listener.d.ts.map +1 -0
  91. package/dist/webhooks/listener.js +140 -0
  92. package/dist/webhooks/listener.js.map +1 -0
  93. package/dist/webhooks/ngrok.d.ts +8 -0
  94. package/dist/webhooks/ngrok.d.ts.map +1 -0
  95. package/dist/webhooks/ngrok.js +50 -0
  96. package/dist/webhooks/ngrok.js.map +1 -0
  97. package/package.json +68 -0
@@ -0,0 +1,250 @@
1
+ import { z } from "zod";
2
+ import { TOOLS } from "../tools/definitions.js";
3
+ // Base schemas
4
+ const StoreIdSchema = z.number().int().positive();
5
+ const CustomerIdSchema = z.number().int().positive();
6
+ const ProductIdSchema = z.number().int().positive();
7
+ const VariantIdSchema = z.number().int().positive();
8
+ const OrderIdSchema = z.number().int().positive();
9
+ const SubscriptionIdSchema = z.number().int().positive();
10
+ const PageSchema = z.number().int().positive().optional();
11
+ const EmailSchema = z.string().email();
12
+ const NonEmptyStringSchema = z.string().min(1);
13
+ // Tool-specific schemas
14
+ export const ToolSchemas = {
15
+ [TOOLS.GET_STORE]: z.object({
16
+ storeId: StoreIdSchema,
17
+ }),
18
+ [TOOLS.LIST_STORES]: z.object({
19
+ page: PageSchema,
20
+ }),
21
+ [TOOLS.GET_CUSTOMER]: z.object({
22
+ customerId: CustomerIdSchema,
23
+ }),
24
+ [TOOLS.LIST_CUSTOMERS]: z.object({
25
+ email: z.string().email().optional(),
26
+ page: PageSchema,
27
+ }),
28
+ [TOOLS.CREATE_CUSTOMER]: z.object({
29
+ storeId: StoreIdSchema,
30
+ name: NonEmptyStringSchema,
31
+ email: EmailSchema,
32
+ city: z.string().optional(),
33
+ country: z.string().optional(),
34
+ region: z.string().optional(),
35
+ }),
36
+ [TOOLS.UPDATE_CUSTOMER]: z.object({
37
+ customerId: CustomerIdSchema,
38
+ name: z.string().optional(),
39
+ email: z.string().email().optional(),
40
+ city: z.string().optional(),
41
+ country: z.string().optional(),
42
+ region: z.string().optional(),
43
+ }),
44
+ [TOOLS.ARCHIVE_CUSTOMER]: z.object({
45
+ customerId: CustomerIdSchema,
46
+ }),
47
+ [TOOLS.GET_PRODUCT]: z.object({
48
+ productId: ProductIdSchema,
49
+ }),
50
+ [TOOLS.LIST_PRODUCTS]: z.object({
51
+ storeId: StoreIdSchema.optional(),
52
+ page: PageSchema,
53
+ }),
54
+ [TOOLS.GET_VARIANT]: z.object({
55
+ variantId: VariantIdSchema,
56
+ }),
57
+ [TOOLS.LIST_VARIANTS]: z.object({
58
+ productId: ProductIdSchema.optional(),
59
+ page: PageSchema,
60
+ }),
61
+ [TOOLS.GET_ORDER]: z.object({
62
+ orderId: OrderIdSchema,
63
+ }),
64
+ [TOOLS.LIST_ORDERS]: z.object({
65
+ page: PageSchema,
66
+ storeId: StoreIdSchema.optional(),
67
+ }),
68
+ [TOOLS.SEARCH_ORDERS]: z.object({
69
+ userEmail: EmailSchema,
70
+ }),
71
+ [TOOLS.GET_ORDER_ITEM]: z.object({
72
+ orderItemId: z.number().int().positive(),
73
+ }),
74
+ [TOOLS.LIST_ORDER_ITEMS]: z.object({
75
+ orderId: OrderIdSchema.optional(),
76
+ page: PageSchema,
77
+ }),
78
+ [TOOLS.GENERATE_ORDER_INVOICE]: z.object({
79
+ orderId: OrderIdSchema,
80
+ name: z.string().optional(),
81
+ email: z.string().email().optional(),
82
+ address: z.string().optional(),
83
+ city: z.string().optional(),
84
+ state: z.string().optional(),
85
+ zip: z.string().optional(),
86
+ country: z.string().optional(),
87
+ }),
88
+ [TOOLS.ISSUE_ORDER_REFUND]: z.object({
89
+ orderId: OrderIdSchema,
90
+ amount: z.number().int().nonnegative(),
91
+ }),
92
+ [TOOLS.GET_SUBSCRIPTION]: z.object({
93
+ subscriptionId: SubscriptionIdSchema,
94
+ }),
95
+ [TOOLS.LIST_SUBSCRIPTIONS]: z.object({
96
+ storeId: StoreIdSchema.optional(),
97
+ page: PageSchema,
98
+ }),
99
+ [TOOLS.UPDATE_SUBSCRIPTION]: z.object({
100
+ subscriptionId: SubscriptionIdSchema,
101
+ variantId: VariantIdSchema.optional(),
102
+ productId: ProductIdSchema.optional(),
103
+ billingAnchor: z.number().int().positive().optional(),
104
+ }),
105
+ [TOOLS.CANCEL_SUBSCRIPTION]: z.object({
106
+ subscriptionId: SubscriptionIdSchema,
107
+ }),
108
+ [TOOLS.GET_SUBSCRIPTION_ITEM]: z.object({
109
+ subscriptionItemId: z.number().int().positive(),
110
+ }),
111
+ [TOOLS.LIST_SUBSCRIPTION_ITEMS]: z.object({
112
+ subscriptionId: SubscriptionIdSchema.optional(),
113
+ page: PageSchema,
114
+ }),
115
+ [TOOLS.GET_SUBSCRIPTION_ITEM_USAGE]: z.object({
116
+ subscriptionItemId: z.number().int().positive(),
117
+ }),
118
+ [TOOLS.GET_SUBSCRIPTION_INVOICE]: z.object({
119
+ subscriptionInvoiceId: z.number().int().positive(),
120
+ }),
121
+ [TOOLS.LIST_SUBSCRIPTION_INVOICES]: z.object({
122
+ subscriptionId: SubscriptionIdSchema.optional(),
123
+ page: PageSchema,
124
+ }),
125
+ [TOOLS.GENERATE_SUBSCRIPTION_INVOICE]: z.object({
126
+ subscriptionInvoiceId: z.number().int().positive(),
127
+ }),
128
+ [TOOLS.ISSUE_SUBSCRIPTION_INVOICE_REFUND]: z.object({
129
+ subscriptionInvoiceId: z.number().int().positive(),
130
+ amount: z.number().int().nonnegative(),
131
+ }),
132
+ [TOOLS.GET_DISCOUNT]: z.object({
133
+ discountId: z.number().int().positive(),
134
+ }),
135
+ [TOOLS.LIST_DISCOUNTS]: z.object({
136
+ storeId: StoreIdSchema.optional(),
137
+ page: PageSchema,
138
+ }),
139
+ [TOOLS.CREATE_DISCOUNT]: z.object({
140
+ storeId: StoreIdSchema,
141
+ name: NonEmptyStringSchema,
142
+ code: NonEmptyStringSchema,
143
+ amount: z.number().nonnegative(),
144
+ amountType: z.enum(["percent", "fixed"]),
145
+ duration: z.enum(["once", "forever", "repeating"]),
146
+ durationInMonths: z.number().int().positive().optional(),
147
+ }),
148
+ [TOOLS.DELETE_DISCOUNT]: z.object({
149
+ discountId: z.number().int().positive(),
150
+ }),
151
+ [TOOLS.GET_LICENSE_KEY]: z.object({
152
+ licenseKeyId: z.number().int().positive(),
153
+ }),
154
+ [TOOLS.LIST_LICENSE_KEYS]: z.object({
155
+ storeId: StoreIdSchema.optional(),
156
+ orderId: OrderIdSchema.optional(),
157
+ page: PageSchema,
158
+ }),
159
+ [TOOLS.UPDATE_LICENSE_KEY]: z.object({
160
+ licenseKeyId: z.number().int().positive(),
161
+ status: z.enum(["active", "inactive"]).optional(),
162
+ }),
163
+ [TOOLS.GET_FILE]: z.object({
164
+ fileId: z.number().int().positive(),
165
+ }),
166
+ [TOOLS.LIST_FILES]: z.object({
167
+ productId: ProductIdSchema.optional(),
168
+ variantId: VariantIdSchema.optional(),
169
+ page: PageSchema,
170
+ }),
171
+ [TOOLS.GET_USAGE_RECORD]: z.object({
172
+ usageRecordId: z.number().int().positive(),
173
+ }),
174
+ [TOOLS.LIST_USAGE_RECORDS]: z.object({
175
+ subscriptionItemId: z.number().int().positive().optional(),
176
+ page: PageSchema,
177
+ }),
178
+ [TOOLS.CREATE_USAGE_RECORD]: z.object({
179
+ subscriptionItemId: z.number().int().positive(),
180
+ quantity: z.number().int().nonnegative(),
181
+ action: z.string().min(1),
182
+ }),
183
+ [TOOLS.CREATE_CHECKOUT]: z.object({
184
+ storeId: StoreIdSchema,
185
+ variantId: VariantIdSchema,
186
+ customPrice: z.number().int().nonnegative().optional(),
187
+ productOptions: z.record(z.unknown()).optional(),
188
+ checkoutOptions: z.record(z.unknown()).optional(),
189
+ checkoutData: z.record(z.unknown()).optional(),
190
+ preview: z.boolean().optional(),
191
+ expiresAt: z.string().datetime().optional(),
192
+ }),
193
+ [TOOLS.GET_WEBHOOK]: z.object({
194
+ webhookId: z.number().int().positive(),
195
+ }),
196
+ [TOOLS.LIST_WEBHOOKS]: z.object({
197
+ storeId: StoreIdSchema.optional(),
198
+ page: PageSchema,
199
+ }),
200
+ [TOOLS.CREATE_WEBHOOK]: z.object({
201
+ storeId: StoreIdSchema,
202
+ url: z.string().url(),
203
+ events: z.array(z.string().min(1)).min(1),
204
+ secret: z.string().min(6).max(40).optional(),
205
+ }),
206
+ [TOOLS.UPDATE_WEBHOOK]: z.object({
207
+ webhookId: z.number().int().positive(),
208
+ url: z.string().url().optional(),
209
+ events: z.array(z.string().min(1)).optional(),
210
+ secret: z.string().min(6).max(40).optional(),
211
+ }),
212
+ [TOOLS.DELETE_WEBHOOK]: z.object({
213
+ webhookId: z.number().int().positive(),
214
+ }),
215
+ [TOOLS.SYNC_CUSTOMER_TO_CRM]: z.object({
216
+ email: EmailSchema,
217
+ name: NonEmptyStringSchema,
218
+ revenue: z.number().nonnegative().optional(),
219
+ company: z.string().optional(),
220
+ title: z.string().optional(),
221
+ }),
222
+ [TOOLS.SEARCH_TRANSACTIONS_NATURAL]: z.object({
223
+ query: NonEmptyStringSchema,
224
+ }),
225
+ [TOOLS.ANALYZE_CHURN_RISK]: z.object({}),
226
+ [TOOLS.CANONIZE_DECISION]: z.object({
227
+ decision: NonEmptyStringSchema,
228
+ rationale: NonEmptyStringSchema,
229
+ expectedOutcome: NonEmptyStringSchema,
230
+ category: z.string().optional(),
231
+ }),
232
+ [TOOLS.GET_FINANCIAL_METRICS]: z.object({
233
+ storeId: StoreIdSchema.optional(),
234
+ startDate: z.string().datetime().optional(),
235
+ endDate: z.string().datetime().optional(),
236
+ }),
237
+ [TOOLS.PREDICT_CHURN_RISK]: z.object({
238
+ storeId: StoreIdSchema.optional(),
239
+ minRiskScore: z.number().int().min(0).max(100).optional(),
240
+ limit: z.number().int().positive().optional(),
241
+ }),
242
+ };
243
+ export function validateToolArgs(toolName, args) {
244
+ const schema = ToolSchemas[toolName];
245
+ if (!schema) {
246
+ throw new Error(`No validation schema found for tool: ${toolName}`);
247
+ }
248
+ return schema.parse(args);
249
+ }
250
+ //# sourceMappingURL=validation.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validation.js","sourceRoot":"","sources":["../../src/utils/validation.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,17 @@
1
+ import { FastifyInstance } from "fastify";
2
+ export interface WebhookPayload {
3
+ meta: {
4
+ event_name: string;
5
+ custom_data?: Record<string, unknown>;
6
+ };
7
+ data: {
8
+ type: string;
9
+ id: string;
10
+ attributes: Record<string, unknown>;
11
+ relationships?: Record<string, unknown>;
12
+ };
13
+ }
14
+ export declare function startWebhookServer(): Promise<FastifyInstance>;
15
+ export declare function stopWebhookServer(): Promise<void>;
16
+ export declare function getWebhookUrl(): string;
17
+ //# sourceMappingURL=listener.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"listener.d.ts","sourceRoot":"","sources":["../../src/webhooks/listener.ts"],"names":[],"mappings":"AAAA,OAAgB,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAQnD,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE;QACJ,UAAU,EAAE,MAAM,CAAC;QACnB,WAAW,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;KACvC,CAAC;IACF,IAAI,EAAE;QACJ,IAAI,EAAE,MAAM,CAAC;QACb,EAAE,EAAE,MAAM,CAAC;QACX,UAAU,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;QACpC,aAAa,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;KACzC,CAAC;CACH;AA0DD,wBAAsB,kBAAkB,IAAI,OAAO,CAAC,eAAe,CAAC,CA4FnE;AAED,wBAAsB,iBAAiB,IAAI,OAAO,CAAC,IAAI,CAAC,CAKvD;AAED,wBAAgB,aAAa,IAAI,MAAM,CAEtC"}
@@ -0,0 +1,140 @@
1
+ import Fastify from "fastify";
2
+ import { createHmac, timingSafeEqual } from "crypto";
3
+ import { config } from "../config.js";
4
+ import { addPaymentEvent } from "../resources/payment-context.js";
5
+ import { logger } from "../utils/logger.js";
6
+ let webhookServer = null;
7
+ function verifyWebhookSignature(rawBody, signature, secret) {
8
+ if (!secret) {
9
+ return false;
10
+ }
11
+ const hmac = createHmac("sha256", secret).update(rawBody).digest("hex");
12
+ const signatureBuffer = Buffer.from(signature, "hex");
13
+ const hmacBuffer = Buffer.from(hmac, "hex");
14
+ if (signatureBuffer.length !== hmacBuffer.length) {
15
+ return false;
16
+ }
17
+ return timingSafeEqual(signatureBuffer, hmacBuffer);
18
+ }
19
+ function parseWebhookEvent(payload) {
20
+ const eventName = payload.meta.event_name;
21
+ const attributes = payload.data.attributes;
22
+ const id = payload.data.id;
23
+ // Extract common fields based on event type
24
+ const orderId = attributes.order_id
25
+ ? parseInt(String(attributes.order_id), 10)
26
+ : attributes.order_number
27
+ ? parseInt(String(attributes.order_number), 10)
28
+ : parseInt(id, 10);
29
+ const customerEmail = (attributes.user_email || attributes.email);
30
+ const status = (attributes.status || attributes.status_formatted);
31
+ const amount = (attributes.total_formatted || attributes.total);
32
+ // Build descriptive message
33
+ const resourceType = payload.data.type.replace("_", " ").replace(/\b\w/g, (l) => l.toUpperCase());
34
+ let message = `${eventName}: ${resourceType} #${id}`;
35
+ if (status)
36
+ message += ` - ${status}`;
37
+ if (amount)
38
+ message += ` - ${amount}`;
39
+ if (customerEmail)
40
+ message += ` (${customerEmail})`;
41
+ return {
42
+ eventName,
43
+ orderId: isNaN(orderId) ? undefined : orderId,
44
+ customerEmail,
45
+ status: status?.toLowerCase(),
46
+ amount,
47
+ message,
48
+ };
49
+ }
50
+ export async function startWebhookServer() {
51
+ if (webhookServer) {
52
+ return webhookServer;
53
+ }
54
+ const fastify = Fastify({
55
+ logger: false,
56
+ bodyLimit: 1048576, // 1MB
57
+ });
58
+ // Decorate request with rawBody
59
+ fastify.decorateRequest("rawBody", "");
60
+ // Capture raw body before JSON parsing
61
+ fastify.addContentTypeParser("application/json", { parseAs: "buffer", bodyLimit: 1048576 }, (req, body, done) => {
62
+ if (body instanceof Buffer) {
63
+ // Store raw body as string for signature verification
64
+ const rawBodyStr = body.toString("utf-8");
65
+ req.rawBody = rawBodyStr;
66
+ try {
67
+ const json = JSON.parse(rawBodyStr);
68
+ done(null, json);
69
+ }
70
+ catch (err) {
71
+ done(err, undefined);
72
+ }
73
+ }
74
+ else {
75
+ done(new Error("Expected Buffer"), undefined);
76
+ }
77
+ });
78
+ // Webhook endpoint
79
+ fastify.post("/webhooks", async (request, reply) => {
80
+ const requestWithRawBody = request;
81
+ const rawBody = requestWithRawBody.rawBody || JSON.stringify(request.body);
82
+ const signature = request.headers["x-signature"] || "";
83
+ const eventName = request.headers["x-event-name"] || "";
84
+ // Verify signature if secret is configured
85
+ if (config.webhookSecret) {
86
+ if (!verifyWebhookSignature(rawBody, signature, config.webhookSecret)) {
87
+ reply.code(401).send({ error: "Invalid signature" });
88
+ return;
89
+ }
90
+ }
91
+ // Parse webhook payload
92
+ let payload;
93
+ try {
94
+ payload = JSON.parse(rawBody);
95
+ }
96
+ catch (error) {
97
+ reply.code(400).send({ error: "Invalid JSON payload" });
98
+ return;
99
+ }
100
+ // Extract event information
101
+ const event = parseWebhookEvent(payload);
102
+ // Add to payment context
103
+ addPaymentEvent({
104
+ timestamp: new Date().toISOString(),
105
+ type: "webhook",
106
+ orderId: event.orderId,
107
+ customerEmail: event.customerEmail,
108
+ status: event.status,
109
+ amount: event.amount,
110
+ message: event.message,
111
+ });
112
+ logger.info({ eventName, orderId: event.orderId, customerEmail: event.customerEmail }, "Webhook received");
113
+ // Return 200 to acknowledge receipt
114
+ reply.code(200).send({ received: true, event: eventName });
115
+ });
116
+ // Health check endpoint
117
+ fastify.get("/health", async () => {
118
+ return { status: "ok", service: "lemonsqueezy-webhook-listener" };
119
+ });
120
+ try {
121
+ await fastify.listen({ port: config.webhookPort, host: "0.0.0.0" });
122
+ logger.info({ port: config.webhookPort }, "Webhook server listening");
123
+ webhookServer = fastify;
124
+ return fastify;
125
+ }
126
+ catch (error) {
127
+ logger.error({ error, port: config.webhookPort }, "Failed to start webhook server");
128
+ throw error;
129
+ }
130
+ }
131
+ export async function stopWebhookServer() {
132
+ if (webhookServer) {
133
+ await webhookServer.close();
134
+ webhookServer = null;
135
+ }
136
+ }
137
+ export function getWebhookUrl() {
138
+ return `http://localhost:${config.webhookPort}/webhooks`;
139
+ }
140
+ //# sourceMappingURL=listener.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"listener.js","sourceRoot":"","sources":["../../src/webhooks/listener.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ export interface NgrokTunnel {
2
+ publicUrl: string;
3
+ localUrl: string;
4
+ }
5
+ export declare function startNgrokTunnel(port: number): Promise<NgrokTunnel>;
6
+ export declare function stopNgrokTunnel(): Promise<void>;
7
+ export declare function getNgrokUrl(): string | null;
8
+ //# sourceMappingURL=ngrok.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ngrok.d.ts","sourceRoot":"","sources":["../../src/webhooks/ngrok.ts"],"names":[],"mappings":"AAOA,MAAM,WAAW,WAAW;IAC1B,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,MAAM,CAAC;CAClB;AAKD,wBAAsB,gBAAgB,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,WAAW,CAAC,CA4CzE;AAED,wBAAsB,eAAe,IAAI,OAAO,CAAC,IAAI,CAAC,CAMrD;AAED,wBAAgB,WAAW,IAAI,MAAM,GAAG,IAAI,CAE3C"}
@@ -0,0 +1,50 @@
1
+ import { exec } from "child_process";
2
+ import { promisify } from "util";
3
+ import { config } from "../config.js";
4
+ import { logger } from "../utils/logger.js";
5
+ const execAsync = promisify(exec);
6
+ let ngrokProcess = null;
7
+ let tunnelInfo = null;
8
+ export async function startNgrokTunnel(port) {
9
+ if (!config.enableNgrok) {
10
+ throw new Error("Ngrok is not enabled in config");
11
+ }
12
+ if (tunnelInfo) {
13
+ return tunnelInfo;
14
+ }
15
+ try {
16
+ // Start ngrok tunnel
17
+ const command = `ngrok http ${port} --log=stdout`;
18
+ ngrokProcess = exec(command);
19
+ // Wait a moment for ngrok to start
20
+ await new Promise((resolve) => setTimeout(resolve, 2000));
21
+ // Get ngrok API URL (ngrok exposes local API on port 4040)
22
+ const response = await fetch("http://localhost:4040/api/tunnels");
23
+ const data = (await response.json());
24
+ if (data.tunnels && data.tunnels.length > 0) {
25
+ const tunnel = data.tunnels[0];
26
+ tunnelInfo = {
27
+ publicUrl: tunnel.public_url,
28
+ localUrl: tunnel.config.addr,
29
+ };
30
+ logger.info({ publicUrl: tunnelInfo.publicUrl, localUrl: tunnelInfo.localUrl }, "Ngrok tunnel established");
31
+ return tunnelInfo;
32
+ }
33
+ throw new Error("Failed to get ngrok tunnel information");
34
+ }
35
+ catch (error) {
36
+ logger.error({ error, port }, "Failed to start ngrok tunnel");
37
+ throw error;
38
+ }
39
+ }
40
+ export async function stopNgrokTunnel() {
41
+ if (ngrokProcess) {
42
+ ngrokProcess.kill();
43
+ ngrokProcess = null;
44
+ tunnelInfo = null;
45
+ }
46
+ }
47
+ export function getNgrokUrl() {
48
+ return tunnelInfo?.publicUrl || null;
49
+ }
50
+ //# sourceMappingURL=ngrok.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ngrok.js","sourceRoot":"","sources":["../../src/webhooks/ngrok.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,eAAe,CAAC;AACrC,OAAO,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;AACjC,OAAO,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AACtC,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAE5C,MAAM,SAAS,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;AAOlC,IAAI,YAAY,GAAmC,IAAI,CAAC;AACxD,IAAI,UAAU,GAAuB,IAAI,CAAC;AAE1C,MAAM,CAAC,KAAK,UAAU,gBAAgB,CAAC,IAAY;IACjD,IAAI,CAAC,MAAM,CAAC,WAAW,EAAE,CAAC;QACxB,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;IACpD,CAAC;IAED,IAAI,UAAU,EAAE,CAAC;QACf,OAAO,UAAU,CAAC;IACpB,CAAC;IAED,IAAI,CAAC;QACH,qBAAqB;QACrB,MAAM,OAAO,GAAG,cAAc,IAAI,eAAe,CAAC;QAClD,YAAY,GAAG,IAAI,CAAC,OAAO,CAAC,CAAC;QAE7B,mCAAmC;QACnC,MAAM,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC,CAAC;QAE1D,2DAA2D;QAC3D,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,mCAAmC,CAAC,CAAC;QAClE,MAAM,IAAI,GAAG,CAAC,MAAM,QAAQ,CAAC,IAAI,EAAE,CAKlC,CAAC;QAEF,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC5C,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;YAC/B,UAAU,GAAG;gBACX,SAAS,EAAE,MAAM,CAAC,UAAU;gBAC5B,QAAQ,EAAE,MAAM,CAAC,MAAM,CAAC,IAAI;aAC7B,CAAC;YACF,MAAM,CAAC,IAAI,CACT,EAAE,SAAS,EAAE,UAAU,CAAC,SAAS,EAAE,QAAQ,EAAE,UAAU,CAAC,QAAQ,EAAE,EAClE,0BAA0B,CAC3B,CAAC;YACF,OAAO,UAAU,CAAC;QACpB,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,wCAAwC,CAAC,CAAC;IAC5D,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,MAAM,CAAC,KAAK,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE,8BAA8B,CAAC,CAAC;QAC9D,MAAM,KAAK,CAAC;IACd,CAAC;AACH,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,eAAe;IACnC,IAAI,YAAY,EAAE,CAAC;QACjB,YAAY,CAAC,IAAI,EAAE,CAAC;QACpB,YAAY,GAAG,IAAI,CAAC;QACpB,UAAU,GAAG,IAAI,CAAC;IACpB,CAAC;AACH,CAAC;AAED,MAAM,UAAU,WAAW;IACzB,OAAO,UAAU,EAAE,SAAS,IAAI,IAAI,CAAC;AACvC,CAAC"}
package/package.json ADDED
@@ -0,0 +1,68 @@
1
+ {
2
+ "name": "lemonsqueezy-mcp-server",
3
+ "version": "1.0.0",
4
+ "mcpName": "io.github.MichaelWeed/lemonsqueezy-mcp-server",
5
+ "description": "Universal Semantic Bridge - Lemon Squeezy SDK as MCP Server with optional Salesforce CRM integration",
6
+ "repository": {
7
+ "type": "git",
8
+ "url": "https://github.com/MichaelWeed/lemonsqueezy-mcp-server.git"
9
+ },
10
+ "type": "module",
11
+ "main": "dist/index.js",
12
+ "bin": {
13
+ "lemonsqueezy-mcp": "./dist/index.js"
14
+ },
15
+ "files": [
16
+ "dist",
17
+ "README.md",
18
+ "SETUP.md",
19
+ "ENV_EXAMPLE.md"
20
+ ],
21
+ "scripts": {
22
+ "build": "tsc",
23
+ "prepublishOnly": "npm run build",
24
+ "start": "node dist/index.js",
25
+ "dev": "tsx src/index.ts",
26
+ "watch": "tsc --watch",
27
+ "test": "vitest run",
28
+ "test:watch": "vitest",
29
+ "test:ui": "vitest --ui"
30
+ },
31
+ "keywords": [
32
+ "mcp",
33
+ "model-context-protocol",
34
+ "lemonsqueezy",
35
+ "payment",
36
+ "sdk",
37
+ "bridge",
38
+ "salesforce",
39
+ "crm",
40
+ "integration"
41
+ ],
42
+ "author": "",
43
+ "license": "MIT",
44
+ "dependencies": {
45
+ "@aws-sdk/client-secrets-manager": "^3.700.0",
46
+ "@lemonsqueezy/lemonsqueezy.js": "^4.0.0",
47
+ "@modelcontextprotocol/sdk": "^1.0.0",
48
+ "dotenv": "^16.4.5",
49
+ "fastify": "^5.6.2",
50
+ "firebase-admin": "^13.6.0",
51
+ "jsforce": "^3.10.10",
52
+ "jsonwebtoken": "^9.0.2",
53
+ "pino": "^10.2.0",
54
+ "zod": "^3.23.8"
55
+ },
56
+ "devDependencies": {
57
+ "@types/jsonwebtoken": "^9.0.5",
58
+ "@types/node": "^20.19.29",
59
+ "@vitest/ui": "^4.0.17",
60
+ "pino-pretty": "^13.1.3",
61
+ "tsx": "^4.7.0",
62
+ "typescript": "^5.3.3",
63
+ "vitest": "^4.0.17"
64
+ },
65
+ "engines": {
66
+ "node": ">=18.0.0"
67
+ }
68
+ }