medusa-plugin-braintree 0.1.1

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.
@@ -0,0 +1,22 @@
1
+ "use strict";
2
+ const widgetModule = { widgets: [] };
3
+ const routeModule = {
4
+ routes: []
5
+ };
6
+ const menuItemModule = {
7
+ menuItems: []
8
+ };
9
+ const formModule = { customFields: {} };
10
+ const displayModule = {
11
+ displays: {}
12
+ };
13
+ const i18nModule = { resources: {} };
14
+ const plugin = {
15
+ widgetModule,
16
+ routeModule,
17
+ menuItemModule,
18
+ formModule,
19
+ displayModule,
20
+ i18nModule
21
+ };
22
+ module.exports = plugin;
@@ -0,0 +1,23 @@
1
+ const widgetModule = { widgets: [] };
2
+ const routeModule = {
3
+ routes: []
4
+ };
5
+ const menuItemModule = {
6
+ menuItems: []
7
+ };
8
+ const formModule = { customFields: {} };
9
+ const displayModule = {
10
+ displays: {}
11
+ };
12
+ const i18nModule = { resources: {} };
13
+ const plugin = {
14
+ widgetModule,
15
+ routeModule,
16
+ menuItemModule,
17
+ formModule,
18
+ displayModule,
19
+ i18nModule
20
+ };
21
+ export {
22
+ plugin as default
23
+ };
@@ -0,0 +1,8 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const utils_1 = require("@medusajs/framework/utils");
4
+ const provider_1 = require("./provider");
5
+ exports.default = (0, utils_1.ModuleProvider)(utils_1.Modules.PAYMENT, {
6
+ services: [provider_1.BraintreeProvider]
7
+ });
8
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvcHJvdmlkZXJzL3BheW1lbnQtYnJhaW50cmVlL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBQUEscURBQW1FO0FBQ25FLHlDQUE4QztBQUU5QyxrQkFBZSxJQUFBLHNCQUFjLEVBQUMsZUFBTyxDQUFDLE9BQU8sRUFBRTtJQUM5QyxRQUFRLEVBQUUsQ0FBQyw0QkFBaUIsQ0FBQztDQUM3QixDQUFDLENBQUEifQ==
@@ -0,0 +1,654 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.BraintreeProvider = exports.buildBraintreeError = void 0;
7
+ const utils_1 = require("@medusajs/framework/utils");
8
+ const braintree_1 = __importDefault(require("braintree"));
9
+ const zod_1 = require("zod");
10
+ const utils_2 = require("./utils");
11
+ const buildTokenCacheKey = (customerId) => `braintree:clientToken:${customerId}`;
12
+ // Type guard utilities for safe type validation
13
+ const validateString = (value, fieldName) => {
14
+ if (typeof value !== 'string' || !value.trim()) {
15
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_ARGUMENT, `${fieldName} must be a non-empty string`);
16
+ }
17
+ return value;
18
+ };
19
+ // Error handling utility that preserves full error context
20
+ const buildBraintreeError = (error, operation, logger, context) => {
21
+ const errorMessage = error instanceof Error ? error.message : String(error);
22
+ // Preserve full error context in logging
23
+ logger.error(`Braintree ${operation} failed: ${errorMessage}`, error instanceof Error ? error : undefined);
24
+ return new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_DATA, `Failed to ${operation}: ${errorMessage}`);
25
+ };
26
+ exports.buildBraintreeError = buildBraintreeError;
27
+ class BraintreeProvider extends utils_1.AbstractPaymentProvider {
28
+ constructor(container, options) {
29
+ super(container, options);
30
+ this.options_ = options;
31
+ this.logger = container[utils_1.ContainerRegistrationKeys.LOGGER];
32
+ this.caching = container[utils_1.Modules.CACHING];
33
+ this.init();
34
+ }
35
+ async saveClientTokenToCache(clientToken, customerId, expiresOnEpochSeconds) {
36
+ if (!customerId)
37
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_ARGUMENT, 'Customer ID is required');
38
+ const nowSeconds = Math.floor(Date.now() / 1000);
39
+ const ttlSeconds = expiresOnEpochSeconds - nowSeconds - 1;
40
+ if (!customerId || !clientToken || ttlSeconds <= 0)
41
+ return;
42
+ await this.caching.set({
43
+ key: buildTokenCacheKey(customerId),
44
+ data: { token: clientToken },
45
+ ttl: ttlSeconds
46
+ });
47
+ }
48
+ async getClientTokenFromCache(customerId) {
49
+ const cached = (await this.caching.get({ key: buildTokenCacheKey(customerId) }));
50
+ return cached?.token ?? null;
51
+ }
52
+ async getValidClientToken(medusaCustomerId, accountHolder) {
53
+ if (!medusaCustomerId) {
54
+ const generatedToken = await this.gateway.clientToken.generate({});
55
+ return generatedToken.clientToken;
56
+ }
57
+ const token = await this.getClientTokenFromCache(medusaCustomerId);
58
+ if (token)
59
+ return token;
60
+ const generatedToken = await this.gateway.clientToken.generate({});
61
+ const defaultExpiryEpochSeconds = Math.floor(Date.now() / 1000) + 24 * 3600; // 24 hours default
62
+ await this.saveClientTokenToCache(generatedToken.clientToken, medusaCustomerId, defaultExpiryEpochSeconds);
63
+ return generatedToken.clientToken;
64
+ }
65
+ async parsePaymentSessionData(data) {
66
+ const schema = zod_1.z.object({
67
+ clientToken: zod_1.z.string().optional(),
68
+ client_token: zod_1.z.string().optional(),
69
+ amount: zod_1.z.number().optional(),
70
+ currency_code: zod_1.z.string().optional(),
71
+ paymentMethodNonce: zod_1.z.string().optional(),
72
+ payment_method_nonce: zod_1.z.string().optional(),
73
+ braintreeTransaction: zod_1.z.any().optional(),
74
+ transaction: zod_1.z.any().optional(),
75
+ account_holder: zod_1.z.any().optional(),
76
+ context: zod_1.z
77
+ .object({
78
+ orderId: zod_1.z.string().optional(),
79
+ customerId: zod_1.z.string().optional(),
80
+ customer: zod_1.z
81
+ .object({
82
+ firstName: zod_1.z.string(),
83
+ lastName: zod_1.z.string(),
84
+ email: zod_1.z.string().email(),
85
+ phone: zod_1.z.string().optional()
86
+ })
87
+ .optional(),
88
+ shipping: zod_1.z
89
+ .object({
90
+ firstName: zod_1.z.string(),
91
+ lastName: zod_1.z.string(),
92
+ streetAddress: zod_1.z.string(),
93
+ extendedAddress: zod_1.z.string().optional(),
94
+ locality: zod_1.z.string(),
95
+ region: zod_1.z.string(),
96
+ postalCode: zod_1.z.string(),
97
+ countryCodeAlpha2: zod_1.z.string()
98
+ })
99
+ .optional(),
100
+ billing: zod_1.z
101
+ .object({
102
+ firstName: zod_1.z.string(),
103
+ lastName: zod_1.z.string(),
104
+ streetAddress: zod_1.z.string(),
105
+ extendedAddress: zod_1.z.string().optional(),
106
+ locality: zod_1.z.string(),
107
+ region: zod_1.z.string(),
108
+ postalCode: zod_1.z.string(),
109
+ countryCodeAlpha2: zod_1.z.string()
110
+ })
111
+ .optional(),
112
+ deviceData: zod_1.z.string().optional(),
113
+ customFields: zod_1.z.any().optional()
114
+ })
115
+ .optional()
116
+ });
117
+ const result = schema.safeParse(data);
118
+ if (!result.success) {
119
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_ARGUMENT, result.error.message);
120
+ }
121
+ result.data.transaction = result.data.transaction ?? result.data.braintreeTransaction;
122
+ result.data.client_token = result.data.client_token ?? result.data.clientToken;
123
+ result.data.payment_method_nonce =
124
+ result.data.payment_method_nonce ?? result.data.paymentMethodNonce;
125
+ return result.data;
126
+ }
127
+ init() {
128
+ const envKey = (this.options_.environment || 'sandbox').toLowerCase();
129
+ const envMap = {
130
+ qa: braintree_1.default.Environment.Qa,
131
+ sandbox: braintree_1.default.Environment.Sandbox,
132
+ production: braintree_1.default.Environment.Production,
133
+ development: braintree_1.default.Environment.Development
134
+ };
135
+ const environment = envMap[envKey] ?? braintree_1.default.Environment.Sandbox;
136
+ this.gateway =
137
+ this.gateway ||
138
+ new braintree_1.default.BraintreeGateway({
139
+ environment,
140
+ merchantId: this.options_.merchantId,
141
+ publicKey: this.options_.publicKey,
142
+ privateKey: this.options_.privateKey
143
+ });
144
+ }
145
+ static validateOptions(options) {
146
+ const requiredStringFields = [
147
+ 'merchantId',
148
+ 'publicKey',
149
+ 'privateKey',
150
+ 'webhookSecret',
151
+ 'environment'
152
+ ];
153
+ for (const field of requiredStringFields) {
154
+ if (!(0, utils_1.isDefined)(options[field]) || typeof options[field] !== 'string') {
155
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_ARGUMENT, `Required option "${field}" is missing or invalid in Braintree plugin`);
156
+ }
157
+ }
158
+ const validEnvironments = ['qa', 'sandbox', 'production', 'development'];
159
+ if (!validEnvironments.includes(options.environment.toLowerCase())) {
160
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_ARGUMENT, `Invalid environment "${options.environment}" in Braintree plugin. Must be one of: ${validEnvironments.join(', ')}`);
161
+ }
162
+ options.enable3DSecure = options.enable3DSecure ?? false;
163
+ options.savePaymentMethod = options.savePaymentMethod ?? false;
164
+ options.autoCapture = options.autoCapture ?? false;
165
+ options.allowRefundOnRefunded = options.allowRefundOnRefunded ?? false;
166
+ const booleanFields = [
167
+ 'enable3DSecure',
168
+ 'savePaymentMethod',
169
+ 'autoCapture',
170
+ 'allowRefundOnRefunded'
171
+ ];
172
+ for (const field of booleanFields) {
173
+ if ((0, utils_1.isDefined)(options[field]) && typeof options[field] !== 'boolean') {
174
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_ARGUMENT, `Option "${field}" must be a boolean in Braintree plugin`);
175
+ }
176
+ }
177
+ }
178
+ async capturePayment(input) {
179
+ const sessionData = await this.parsePaymentSessionData(input.data ?? {});
180
+ const transaction = sessionData.transaction;
181
+ if (!transaction) {
182
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.NOT_FOUND, 'Braintree transaction not found');
183
+ }
184
+ const paymentsResponse = await this.gateway.transaction.find(transaction.id);
185
+ switch (paymentsResponse.status) {
186
+ case 'authorized': {
187
+ const { id, amount } = paymentsResponse;
188
+ const toPay = amount.toString();
189
+ const captureResult = await this.gateway.transaction.submitForSettlement(id, toPay);
190
+ if (captureResult.success) {
191
+ const _transaction = await this.retrieveTransaction(transaction.id);
192
+ return {
193
+ data: {
194
+ ...input.data,
195
+ transaction: _transaction
196
+ }
197
+ };
198
+ }
199
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.NOT_FOUND, `No payments found for transaction ${transaction.id}`);
200
+ }
201
+ case 'settled':
202
+ case 'settling':
203
+ case 'submitted_for_settlement': {
204
+ const _transaction = await this.retrieveTransaction(transaction.id);
205
+ return {
206
+ data: {
207
+ ...input.data,
208
+ transaction: _transaction
209
+ }
210
+ };
211
+ }
212
+ default: {
213
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.NOT_FOUND, `Not in a state to settle ${transaction.id}`);
214
+ }
215
+ }
216
+ }
217
+ async authorizePayment(input) {
218
+ try {
219
+ const sessionData = await this.parsePaymentSessionData(input.data ?? {});
220
+ let transaction = sessionData.transaction;
221
+ if (!sessionData.payment_method_nonce)
222
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_ARGUMENT, 'Payment method nonce is required');
223
+ if (!transaction) {
224
+ transaction = await this.createTransaction({
225
+ input
226
+ });
227
+ }
228
+ const paymentStatusRequest = {
229
+ ...input,
230
+ data: {
231
+ ...input.data,
232
+ transaction
233
+ }
234
+ };
235
+ const status = await this.getPaymentStatus(paymentStatusRequest);
236
+ const finalStatus = status.status === 'authorized' && this.options_.autoCapture ? 'captured' : status.status;
237
+ return {
238
+ data: {
239
+ ...input.data,
240
+ transaction
241
+ },
242
+ status: finalStatus
243
+ };
244
+ }
245
+ catch (error) {
246
+ const message = error instanceof Error ? error.message : 'Unknown error';
247
+ this.logger.error(`Error authorizing transaction: ${message}`, error instanceof Error ? error : undefined);
248
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_DATA, message);
249
+ }
250
+ }
251
+ async cancelPayment(input) {
252
+ const sessionData = await this.parsePaymentSessionData(input.data ?? {});
253
+ const transaction = await this.retrieveTransaction(sessionData.transaction?.id);
254
+ if (!transaction)
255
+ return {};
256
+ if (transaction.status === 'authorized' ||
257
+ transaction.status === 'submitted_for_settlement' ||
258
+ (transaction.status === 'settling' &&
259
+ transaction.paymentInstrumentType === 'paypal_account')) {
260
+ const updatedTransaction = await this.gateway.transaction.void(transaction.id);
261
+ if (updatedTransaction) {
262
+ const updated = await this.retrieveTransaction(transaction.id);
263
+ return {
264
+ data: {
265
+ ...input.data,
266
+ transaction: updated
267
+ }
268
+ };
269
+ }
270
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.NOT_FOUND, `No payments found for transaction ${transaction.id}`);
271
+ }
272
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.NOT_FOUND, `Payment cannot be cancelled ${transaction.id}`);
273
+ }
274
+ async retrieveTransaction(id, throwOnMissing = true) {
275
+ const transactionData = await this.gateway.transaction.find(id);
276
+ if (!transactionData && throwOnMissing)
277
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.NOT_FOUND, `Braintree transaction not found: ${id}`);
278
+ return transactionData;
279
+ }
280
+ mapTransactionStatusToPaymentStatus(status) {
281
+ switch (status) {
282
+ case 'authorization_expired':
283
+ return utils_1.PaymentSessionStatus.CANCELED;
284
+ case 'authorizing':
285
+ return utils_1.PaymentSessionStatus.REQUIRES_MORE;
286
+ case 'authorized':
287
+ return utils_1.PaymentSessionStatus.AUTHORIZED;
288
+ case 'settled':
289
+ case 'settlement_confirmed':
290
+ return utils_1.PaymentSessionStatus.CAPTURED;
291
+ case 'settling':
292
+ case 'settlement_pending':
293
+ case 'submitted_for_settlement':
294
+ return utils_1.PaymentSessionStatus.AUTHORIZED;
295
+ case 'voided':
296
+ return utils_1.PaymentSessionStatus.CANCELED;
297
+ case 'failed':
298
+ return utils_1.PaymentSessionStatus.ERROR;
299
+ default:
300
+ return utils_1.PaymentSessionStatus.PENDING;
301
+ }
302
+ }
303
+ validateInitiatePaymentData(data) {
304
+ const schema = zod_1.z.object({
305
+ paymentMethodNonce: zod_1.z.string().optional(),
306
+ payment_method_nonce: zod_1.z.string().optional(),
307
+ cardDetails: zod_1.z
308
+ .object({
309
+ cardType: zod_1.z.string().optional(),
310
+ lastFour: zod_1.z.string().optional(),
311
+ lastTwo: zod_1.z.string().optional(),
312
+ expirationMonth: zod_1.z.string().optional(),
313
+ expirationYear: zod_1.z.string().optional(),
314
+ cardholderName: zod_1.z.string().optional()
315
+ })
316
+ .optional()
317
+ });
318
+ const result = schema.safeParse(data);
319
+ if (!result.success) {
320
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_ARGUMENT, result.error.message);
321
+ }
322
+ result.data.payment_method_nonce =
323
+ result.data.payment_method_nonce ?? result.data.paymentMethodNonce;
324
+ return result.data;
325
+ }
326
+ async initiatePayment(input) {
327
+ const data = this.validateInitiatePaymentData(input.data ?? {});
328
+ let transaction;
329
+ const token = await this.getValidClientToken(input.context?.customer?.id, input.context?.account_holder);
330
+ const paymentSessionId = validateString(input.context?.idempotency_key, 'Payment session ID');
331
+ if (!token) {
332
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_ARGUMENT, 'Failed to generate client token');
333
+ }
334
+ const dataToSave = {
335
+ transaction: transaction,
336
+ client_token: token,
337
+ payment_method_nonce: data?.payment_method_nonce,
338
+ amount: Number(input.amount),
339
+ currency_code: input.currency_code,
340
+ account_holder: input.context?.account_holder
341
+ };
342
+ return {
343
+ id: paymentSessionId,
344
+ data: { ...dataToSave }
345
+ };
346
+ }
347
+ async createTransaction({ input }) {
348
+ const sessionData = await this.parsePaymentSessionData(input.data ?? {});
349
+ const { context } = sessionData;
350
+ let transactionRequest = {
351
+ amount: (0, utils_2.formatToTwoDecimalString)(Number(sessionData.amount)),
352
+ paymentMethodNonce: validateString(sessionData.payment_method_nonce, 'Payment method nonce'),
353
+ options: {
354
+ submitForSettlement: this.options_.autoCapture,
355
+ storeInVaultOnSuccess: this.options_.savePaymentMethod,
356
+ storeInVault: this.options_.savePaymentMethod,
357
+ threeDSecure: this.options_.enable3DSecure
358
+ ? {
359
+ required: this.options_.enable3DSecure
360
+ }
361
+ : undefined
362
+ }
363
+ };
364
+ if (context?.orderId)
365
+ transactionRequest.orderId = context.orderId;
366
+ if (context?.customer)
367
+ transactionRequest.customer = context.customer;
368
+ if (context?.shipping)
369
+ transactionRequest.shipping = context.shipping;
370
+ if (context?.billing)
371
+ transactionRequest.billing = context.billing;
372
+ if (context?.deviceData)
373
+ transactionRequest.deviceData = context.deviceData;
374
+ if (context?.customFields)
375
+ transactionRequest.customFields = context.customFields;
376
+ try {
377
+ const saleResponse = await this.gateway.transaction.sale(transactionRequest);
378
+ if (!saleResponse.success) {
379
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.PAYMENT_AUTHORIZATION_ERROR, saleResponse.transaction?.gatewayRejectionReason ?? 'Unknown error');
380
+ }
381
+ try {
382
+ return await this.retrieveTransaction(saleResponse.transaction.id);
383
+ }
384
+ catch (error) {
385
+ if (saleResponse.transaction?.id) {
386
+ await this.gateway.transaction.void(saleResponse.transaction.id);
387
+ }
388
+ throw (0, exports.buildBraintreeError)(error, 'sync payment session', this.logger, {
389
+ transactionId: saleResponse.transaction?.id
390
+ });
391
+ }
392
+ }
393
+ catch (error) {
394
+ throw (0, exports.buildBraintreeError)(error, 'create Braintree transaction', this.logger);
395
+ }
396
+ }
397
+ async deletePayment(input) {
398
+ const sessionData = await this.parsePaymentSessionData(input.data ?? {});
399
+ const transaction = sessionData.transaction;
400
+ if (transaction) {
401
+ try {
402
+ const result = await this.cancelPayment(input);
403
+ return {
404
+ data: {
405
+ ...input.data,
406
+ transaction: result.data?.transaction
407
+ }
408
+ };
409
+ }
410
+ catch (e) {
411
+ throw (0, exports.buildBraintreeError)(e, 'delete Braintree payment', this.logger);
412
+ }
413
+ }
414
+ else {
415
+ return {
416
+ data: {
417
+ ...input.data,
418
+ transaction
419
+ }
420
+ };
421
+ }
422
+ }
423
+ async getPaymentStatus(input) {
424
+ // Support both `data.transaction` and `data.braintreeTransaction` without requiring full session parsing
425
+ const tx = (input.data?.transaction ?? input.data?.braintreeTransaction);
426
+ const id = tx?.id;
427
+ if (!id) {
428
+ return { status: utils_1.PaymentSessionStatus.PENDING };
429
+ }
430
+ let transaction;
431
+ try {
432
+ transaction = await this.gateway.transaction.find(id);
433
+ }
434
+ catch (e) {
435
+ this.logger.warn('received payment data from session not transaction data');
436
+ throw e;
437
+ }
438
+ const status = this.mapTransactionStatusToPaymentStatus(transaction.status);
439
+ return { status };
440
+ }
441
+ async savePaymentMethod(input) {
442
+ const sessionData = await this.parsePaymentSessionData(input.data ?? {});
443
+ const braintreeCustomerId = validateString(input.context?.account_holder?.data?.id, 'Braintree customer ID');
444
+ const paymentMethodNonce = sessionData?.payment_method_nonce;
445
+ if (!paymentMethodNonce) {
446
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_ARGUMENT, 'Payment method nonce is required');
447
+ }
448
+ const paymentMethodResult = await this.gateway.paymentMethod.create({
449
+ customerId: braintreeCustomerId,
450
+ paymentMethodNonce: paymentMethodNonce
451
+ });
452
+ if (!paymentMethodResult.success) {
453
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_DATA, `Failed to save payment method: ${JSON.stringify(paymentMethodResult.errors)}`);
454
+ }
455
+ return {
456
+ id: paymentMethodResult.paymentMethod.token,
457
+ data: {
458
+ ...input.data
459
+ }
460
+ };
461
+ }
462
+ async refundPayment(input) {
463
+ const sessionData = await this.parsePaymentSessionData(input.data ?? {});
464
+ const refundAmountBN = utils_1.MathBN.convert(input.amount, 2);
465
+ const refundAmount = refundAmountBN.toNumber();
466
+ if (!refundAmount)
467
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_DATA, 'Refund amount is invalid');
468
+ let transaction = await this.retrieveTransaction(sessionData.transaction?.id);
469
+ let shouldVoid = ['submitted_for_settlement', 'authorized'].includes(transaction.status);
470
+ if (process.env.TEST_FORCE_SETTLED === 'true') {
471
+ shouldVoid = false;
472
+ await this.gateway.testing.settle(transaction.id);
473
+ transaction = await this.retrieveTransaction(transaction.id);
474
+ }
475
+ if (shouldVoid) {
476
+ const voidResponse = await this.gateway.transaction.void(transaction.id);
477
+ const voidSucceeded = voidResponse.success ?? false;
478
+ if (!voidSucceeded)
479
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.PAYMENT_AUTHORIZATION_ERROR, 'Failed to void transaction');
480
+ const voidedTransaction = voidResponse?.transaction ?? (await this.retrieveTransaction(transaction.id));
481
+ const refundResult = {
482
+ data: {
483
+ ...input.data,
484
+ transaction: voidedTransaction,
485
+ braintreeRefund: {
486
+ success: true,
487
+ transactionId: voidedTransaction?.id,
488
+ type: 'void'
489
+ }
490
+ }
491
+ };
492
+ return refundResult;
493
+ }
494
+ const shouldRefund = ['settled', 'settling'].includes(transaction.status);
495
+ if (!shouldRefund) {
496
+ this.logger.error(`Braintree transaction with ID ${transaction.id} cannot be refunded because it's in status ${transaction.status}`);
497
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.NOT_FOUND, `Braintree transaction with ID ${transaction.id} cannot be refunded`);
498
+ }
499
+ if (transaction.id) {
500
+ const refundAmountDecimal = (0, utils_2.formatToTwoDecimalString)(refundAmount);
501
+ try {
502
+ this.logger.info(`Refunding transaction: ${transaction.id} with amount: ${refundAmountDecimal} (created from ${refundAmount})`);
503
+ const refundResponse = await this.gateway.transaction.refund(transaction.id, refundAmountDecimal);
504
+ const refundSucceeded = refundResponse.success ?? false;
505
+ if (!refundSucceeded)
506
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_DATA, `Failed to create Braintree refund: ${refundResponse.message}`);
507
+ const refundTransaction = refundResponse.transaction ?? (await this.retrieveTransaction(transaction.id));
508
+ const refundResult = {
509
+ data: {
510
+ ...input.data,
511
+ transaction: refundTransaction,
512
+ braintreeRefund: refundTransaction
513
+ }
514
+ };
515
+ return refundResult;
516
+ }
517
+ catch (e) {
518
+ throw (0, exports.buildBraintreeError)(e, 'create Braintree refund', this.logger);
519
+ }
520
+ }
521
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.NOT_FOUND, `Braintree transaction with ID ${transaction.id} not found`);
522
+ }
523
+ async retrievePayment(input) {
524
+ const paymentSessionData = await this.parsePaymentSessionData(input.data ?? {});
525
+ if (!paymentSessionData.transaction?.id) {
526
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.NOT_FOUND, 'Braintree transaction not found');
527
+ }
528
+ const retrieved = await this.retrieveTransaction(paymentSessionData.transaction?.id);
529
+ return {
530
+ data: {
531
+ ...input.data,
532
+ transaction: retrieved
533
+ }
534
+ };
535
+ }
536
+ async updatePayment(input) {
537
+ return Promise.resolve({
538
+ data: {
539
+ ...input.data,
540
+ amount: input.amount,
541
+ currency_code: input.currency_code
542
+ }
543
+ });
544
+ }
545
+ async createAccountHolder(input) {
546
+ const customer = await this.createBraintreeCustomer(input.context.customer);
547
+ return {
548
+ id: customer.id,
549
+ data: {
550
+ ...customer
551
+ }
552
+ };
553
+ }
554
+ async updateAccountHolder(input) {
555
+ const { context } = input;
556
+ const accountHolderId = context.account_holder?.data?.id;
557
+ if (!accountHolderId) {
558
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_DATA, `Account holder id is required`);
559
+ }
560
+ try {
561
+ const accountHolder = await this.gateway.customer.find(accountHolderId);
562
+ if (!accountHolder) {
563
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.NOT_FOUND, `Account holder with id ${accountHolderId} not found`);
564
+ }
565
+ const customerUpdateRequest = {
566
+ firstName: input.context.customer?.first_name ?? accountHolder.firstName,
567
+ lastName: input.context.customer?.last_name ?? accountHolder.lastName,
568
+ email: input.context.customer?.email ?? accountHolder.email,
569
+ phone: input.context.customer?.phone ?? accountHolder.phone
570
+ };
571
+ const updateResult = await this.gateway.customer.update(accountHolder.id, customerUpdateRequest);
572
+ if (!updateResult.success) {
573
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_DATA, `Failed to update account holder: ${JSON.stringify(updateResult.errors)}`);
574
+ }
575
+ return {
576
+ data: { ...updateResult.customer }
577
+ };
578
+ }
579
+ catch (e) {
580
+ throw (0, exports.buildBraintreeError)(e, 'update account holder', this.logger);
581
+ }
582
+ }
583
+ async deleteAccountHolder(input) {
584
+ const { context } = input;
585
+ const accountHolderId = context.account_holder?.data?.id;
586
+ if (!accountHolderId)
587
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_DATA, `Account holder id is required`);
588
+ try {
589
+ const accountHolder = await this.gateway.customer.find(accountHolderId);
590
+ if (!accountHolder)
591
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.NOT_FOUND, `Account holder with id ${accountHolderId} not found`);
592
+ await this.gateway.customer.delete(accountHolder.id);
593
+ return {
594
+ data: {}
595
+ };
596
+ }
597
+ catch (e) {
598
+ throw (0, exports.buildBraintreeError)(e, 'delete account holder', this.logger);
599
+ }
600
+ }
601
+ async getWebhookActionAndData(webhookData) {
602
+ const logger = this.logger;
603
+ logger.info(`Received Braintree webhook with data: ${!!webhookData.data}`);
604
+ const decodedPayload = new URLSearchParams(webhookData.data);
605
+ let notification;
606
+ try {
607
+ notification = (await this.gateway.webhookNotification.parse(decodedPayload.get('bt_signature') ?? '', decodedPayload.get('bt_payload') ?? ''));
608
+ if (!notification) {
609
+ return { action: utils_1.PaymentActions.FAILED };
610
+ }
611
+ }
612
+ catch (error) {
613
+ logger.error(`Braintree webhook validation failed : ${error}`);
614
+ return { action: utils_1.PaymentActions.FAILED };
615
+ }
616
+ const paymentData = await this.gateway.transaction.find(notification.transaction.id);
617
+ const customFields = paymentData.customFields;
618
+ switch (notification.kind) {
619
+ case 'transaction_settled':
620
+ return {
621
+ action: utils_1.PaymentActions.SUCCESSFUL,
622
+ data: {
623
+ session_id: customFields.medusa_payment_session_id ?? '',
624
+ amount: paymentData.amount
625
+ }
626
+ };
627
+ case 'transaction_settlement_declined':
628
+ return {
629
+ action: utils_1.PaymentActions.FAILED,
630
+ data: {
631
+ session_id: customFields.medusa_payment_session_id ?? '',
632
+ amount: paymentData.amount
633
+ }
634
+ };
635
+ default:
636
+ return { action: utils_1.PaymentActions.NOT_SUPPORTED };
637
+ }
638
+ }
639
+ async createBraintreeCustomer(customer) {
640
+ const customerResult = await this.gateway.customer.create({
641
+ email: customer.email,
642
+ firstName: customer.first_name ?? undefined,
643
+ lastName: customer.last_name ?? undefined,
644
+ phone: customer.phone ?? undefined
645
+ });
646
+ if (!customerResult.success) {
647
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_DATA, `Failed to create Braintree customer: ${JSON.stringify(customerResult.errors)}`);
648
+ }
649
+ return customerResult.customer;
650
+ }
651
+ }
652
+ exports.BraintreeProvider = BraintreeProvider;
653
+ BraintreeProvider.identifier = 'braintree';
654
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvdmlkZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvcHJvdmlkZXJzL3BheW1lbnQtYnJhaW50cmVlL3Byb3ZpZGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7OztBQUFBLHFEQVNrQztBQXFDbEMsMERBQWlDO0FBQ2pDLDZCQUF1QjtBQUN2QixtQ0FBa0Q7QUFpQ2xELE1BQU0sa0JBQWtCLEdBQUcsQ0FBQyxVQUFrQixFQUFFLEVBQUUsQ0FBQyx5QkFBeUIsVUFBVSxFQUFFLENBQUE7QUFFeEYsZ0RBQWdEO0FBQ2hELE1BQU0sY0FBYyxHQUFHLENBQUMsS0FBYyxFQUFFLFNBQWlCLEVBQVUsRUFBRTtJQUNwRSxJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVEsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLEVBQUUsRUFBRSxDQUFDO1FBQ2hELE1BQU0sSUFBSSxtQkFBVyxDQUNwQixtQkFBVyxDQUFDLEtBQUssQ0FBQyxnQkFBZ0IsRUFDbEMsR0FBRyxTQUFTLDZCQUE2QixDQUN6QyxDQUFBO0lBQ0YsQ0FBQztJQUNELE9BQU8sS0FBSyxDQUFBO0FBQ2IsQ0FBQyxDQUFBO0FBRUQsMkRBQTJEO0FBQ3BELE1BQU0sbUJBQW1CLEdBQUcsQ0FDbEMsS0FBYyxFQUNkLFNBQWlCLEVBQ2pCLE1BQWMsRUFDZCxPQUFpQyxFQUNuQixFQUFFO0lBQ2hCLE1BQU0sWUFBWSxHQUFHLEtBQUssWUFBWSxLQUFLLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQTtJQUUzRSx5Q0FBeUM7SUFDekMsTUFBTSxDQUFDLEtBQUssQ0FDWCxhQUFhLFNBQVMsWUFBWSxZQUFZLEVBQUUsRUFDaEQsS0FBSyxZQUFZLEtBQUssQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQzFDLENBQUE7SUFFRCxPQUFPLElBQUksbUJBQVcs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
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvcHJvdmlkZXJzL3BheW1lbnQtYnJhaW50cmVlL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIifQ==
@@ -0,0 +1,23 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.formatToTwoDecimalString = formatToTwoDecimalString;
4
+ const utils_1 = require("@medusajs/framework/utils");
5
+ /**
6
+ * Formats a number or string to a two-decimal string representation.
7
+ * Validates the input is parseable to a number and throws MedusaError on NaN.
8
+ *
9
+ * @param amount - The amount to format (number or string)
10
+ * @returns A string representation with exactly 2 decimal places
11
+ * @throws MedusaError if the amount is not a valid number
12
+ */
13
+ function formatToTwoDecimalString(amount) {
14
+ if (typeof amount !== 'string') {
15
+ amount = amount.toString();
16
+ }
17
+ const num = Number.parseFloat(amount);
18
+ if (Number.isNaN(num)) {
19
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_ARGUMENT, 'Invalid amount');
20
+ }
21
+ return num.toFixed(2);
22
+ }
23
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvcHJvdmlkZXJzL3BheW1lbnQtYnJhaW50cmVlL3V0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBVUEsNERBWUM7QUF0QkQscURBQXVEO0FBRXZEOzs7Ozs7O0dBT0c7QUFDSCxTQUFnQix3QkFBd0IsQ0FBQyxNQUF1QjtJQUMvRCxJQUFJLE9BQU8sTUFBTSxLQUFLLFFBQVEsRUFBRSxDQUFDO1FBQ2hDLE1BQU0sR0FBRyxNQUFNLENBQUMsUUFBUSxFQUFFLENBQUE7SUFDM0IsQ0FBQztJQUVELE1BQU0sR0FBRyxHQUFHLE1BQU0sQ0FBQyxVQUFVLENBQUMsTUFBTSxDQUFDLENBQUE7SUFFckMsSUFBSSxNQUFNLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUM7UUFDdkIsTUFBTSxJQUFJLG1CQUFXLENBQUMsbUJBQVcsQ0FBQyxLQUFLLENBQUMsZ0JBQWdCLEVBQUUsZ0JBQWdCLENBQUMsQ0FBQTtJQUM1RSxDQUFDO0lBRUQsT0FBTyxHQUFHLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFBO0FBQ3RCLENBQUMifQ==
package/README.md ADDED
File without changes
package/package.json ADDED
@@ -0,0 +1,77 @@
1
+ {
2
+ "name": "medusa-plugin-braintree",
3
+ "version": "0.1.1",
4
+ "description": "Braintree payment provider plugin for Medusa v2",
5
+ "author": "Lacey Pevey",
6
+ "license": "MIT",
7
+ "repository": {
8
+ "type": "git",
9
+ "url": "https://github.com/pevey/medusa-plugins"
10
+ },
11
+ "homepage": "https://pevey.com/medusa-plugin-braintree",
12
+ "files": [
13
+ ".medusa/server"
14
+ ],
15
+ "exports": {
16
+ "./package.json": "./package.json",
17
+ "./.medusa/server/src/modules/*": "./.medusa/server/src/modules/*/index.js",
18
+ "./modules/*": "./.medusa/server/src/modules/*/index.js",
19
+ "./providers/*": "./.medusa/server/src/providers/*/index.js",
20
+ "./*": "./.medusa/server/src/*.js",
21
+ ".": "./.medusa/server/src/providers/payment-braintree/index.js"
22
+ },
23
+ "keywords": [
24
+ "medusa-plugin-payment",
25
+ "medusa-v2",
26
+ "medusa",
27
+ "medusa-plugin-braintree",
28
+ "braintree",
29
+ "payment"
30
+ ],
31
+ "scripts": {
32
+ "build": "medusa plugin:build",
33
+ "dev": "medusa plugin:develop",
34
+ "prepublishOnly": "medusa plugin:build",
35
+ "test:unit": "TEST_TYPE=unit NODE_OPTIONS=--experimental-vm-modules jest --silent --runInBand --detectOpenHandles --forceExit"
36
+ },
37
+ "dependencies": {
38
+ "braintree": "3.36.0"
39
+ },
40
+ "devDependencies": {
41
+ "@medusajs/admin-sdk": "2.13.6",
42
+ "@medusajs/cli": "2.13.6",
43
+ "@medusajs/framework": "2.13.6",
44
+ "@medusajs/icons": "2.13.6",
45
+ "@medusajs/medusa": "2.13.6",
46
+ "@medusajs/test-utils": "2.13.6",
47
+ "@medusajs/ui": "4.0.25",
48
+ "@swc/core": "^1.7.28",
49
+ "@swc/jest": "^0.2.39",
50
+ "@types/braintree": "3.4.4",
51
+ "@types/jest": "^30.0.0",
52
+ "@types/node": "^20.0.0",
53
+ "@types/react": "^18.3.2",
54
+ "@types/react-dom": "^18.2.25",
55
+ "jest": "^29.7.0",
56
+ "prop-types": "^15.8.1",
57
+ "react": "^18.2.0",
58
+ "react-dom": "^18.2.0",
59
+ "ts-node": "^10.9.2",
60
+ "typescript": "^5.6.2",
61
+ "vite": "^5.2.11",
62
+ "yalc": "^1.0.0-pre.53"
63
+ },
64
+ "peerDependencies": {
65
+ "@medusajs/admin-sdk": "2.13.6",
66
+ "@medusajs/cli": "2.13.6",
67
+ "@medusajs/framework": "2.13.6",
68
+ "@medusajs/icons": "2.13.6",
69
+ "@medusajs/medusa": "2.13.6",
70
+ "@medusajs/test-utils": "2.13.6",
71
+ "@medusajs/ui": "4.0.25"
72
+ },
73
+ "engines": {
74
+ "node": ">=20"
75
+ },
76
+ "packageManager": "yarn@4.13.0+sha512.5c20ba010c99815433e5c8453112165e673f1c7948d8d2b267f4b5e52097538658388ebc9f9580656d9b75c5cc996f990f611f99304a2197d4c56d21eea370e7"
77
+ }