@jsm406/medusa-plugin-redsys 1.0.3

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,18 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./types"), exports);
18
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLDBDQUF1QiJ9
@@ -0,0 +1,11 @@
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
+ const utils_1 = require("@medusajs/framework/utils");
7
+ const service_1 = __importDefault(require("./service"));
8
+ exports.default = (0, utils_1.ModuleProvider)(utils_1.Modules.PAYMENT, {
9
+ services: [service_1.default],
10
+ });
11
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvcHJvdmlkZXJzL3JlZHN5cy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLHFEQUFtRTtBQUNuRSx3REFBNkM7QUFFN0Msa0JBQWUsSUFBQSxzQkFBYyxFQUFDLGVBQU8sQ0FBQyxPQUFPLEVBQUU7SUFDN0MsUUFBUSxFQUFFLENBQUMsaUJBQXFCLENBQUM7Q0FDbEMsQ0FBQyxDQUFBIn0=
@@ -0,0 +1,331 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const utils_1 = require("@medusajs/framework/utils");
4
+ const redsys_easy_1 = require("redsys-easy");
5
+ const amount_1 = require("../../utils/amount");
6
+ const currency_1 = require("../../utils/currency");
7
+ const order_id_1 = require("../../utils/order-id");
8
+ const errors_1 = require("../../utils/errors");
9
+ const DEFAULTS = {
10
+ terminal: "001",
11
+ transactionType: "0",
12
+ };
13
+ class RedsysProviderService extends utils_1.AbstractPaymentProvider {
14
+ static validateOptions(options) {
15
+ if (!options.secretKey || typeof options.secretKey !== "string") {
16
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_DATA, "Redsys secretKey is required and must be a string");
17
+ }
18
+ if (!options.merchantCode || typeof options.merchantCode !== "string") {
19
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_DATA, "Redsys merchantCode is required and must be a string");
20
+ }
21
+ }
22
+ constructor(container, options) {
23
+ super(container, options);
24
+ this.logger_ = container.logger;
25
+ this.options_ = options;
26
+ this.redsysApi = (0, redsys_easy_1.createRedsysAPI)({
27
+ secretKey: options.secretKey,
28
+ urls: options.environment === "production" ? redsys_easy_1.PRODUCTION_URLS : redsys_easy_1.SANDBOX_URLS,
29
+ });
30
+ }
31
+ // ---------- Initiate ----------
32
+ async initiatePayment(input) {
33
+ const orderId = (0, order_id_1.generateOrderId)();
34
+ const sessionId = "redsys_" + orderId;
35
+ const amount = this.assertPositiveAmount(input.amount);
36
+ const amountStr = String((0, amount_1.getSmallestUnit)(amount, input.currency_code));
37
+ const currencyNum = (0, currency_1.getCurrencyNum)(input.currency_code);
38
+ const transactionType = this.options_.transactionType || DEFAULTS.transactionType;
39
+ const merchantParams = {
40
+ DS_MERCHANT_MERCHANTCODE: this.options_.merchantCode,
41
+ DS_MERCHANT_TERMINAL: this.options_.terminal || DEFAULTS.terminal,
42
+ DS_MERCHANT_ORDER: orderId,
43
+ DS_MERCHANT_AMOUNT: amountStr,
44
+ DS_MERCHANT_CURRENCY: currencyNum,
45
+ DS_MERCHANT_TRANSACTIONTYPE: transactionType,
46
+ DS_MERCHANT_CONSUMERLANGUAGE: "1",
47
+ };
48
+ if (this.options_.notificationUrl) {
49
+ merchantParams.DS_MERCHANT_MERCHANTURL = this.options_.notificationUrl;
50
+ }
51
+ const separator = (url) => url.includes("?") ? "&" : "?";
52
+ if (this.options_.successUrl) {
53
+ merchantParams.DS_MERCHANT_URLOK =
54
+ this.options_.successUrl + separator(this.options_.successUrl) + "orderId=" + orderId;
55
+ }
56
+ if (this.options_.errorUrl) {
57
+ merchantParams.DS_MERCHANT_URLKO =
58
+ this.options_.errorUrl + separator(this.options_.errorUrl) + "orderId=" + orderId;
59
+ }
60
+ merchantParams.DS_MERCHANT_MERCHANTDATA = sessionId + "|" + orderId;
61
+ const form = await this.redsysApi.createRedirectForm(merchantParams);
62
+ const sessionData = {
63
+ orderId,
64
+ amount: amountStr,
65
+ currency: currencyNum,
66
+ status: "pending",
67
+ transactionType,
68
+ merchantParams: form.body.Ds_MerchantParameters,
69
+ signature: form.body.Ds_Signature,
70
+ signatureVersion: form.body.Ds_SignatureVersion,
71
+ formUrl: form.url,
72
+ };
73
+ this.logger_.info("[REDSYS] Redirect form created for order: " + orderId);
74
+ return {
75
+ id: sessionId,
76
+ data: sessionData,
77
+ };
78
+ }
79
+ // ---------- Authorize ----------
80
+ async authorizePayment(input) {
81
+ const sessionData = input.data;
82
+ if (sessionData?.status === "authorized" ||
83
+ sessionData?.status === "pending") {
84
+ return {
85
+ status: utils_1.PaymentSessionStatus.AUTHORIZED,
86
+ data: input.data,
87
+ };
88
+ }
89
+ return {
90
+ status: utils_1.PaymentSessionStatus.PENDING,
91
+ data: input.data,
92
+ };
93
+ }
94
+ // ---------- Capture ----------
95
+ async capturePayment(input) {
96
+ const sessionData = input.data;
97
+ if (!sessionData?.orderId) {
98
+ return { data: input.data };
99
+ }
100
+ const transactionType = sessionData.transactionType || DEFAULTS.transactionType;
101
+ if (transactionType !== "1") {
102
+ return { data: input.data };
103
+ }
104
+ const params = {
105
+ DS_MERCHANT_MERCHANTCODE: this.options_.merchantCode,
106
+ DS_MERCHANT_TERMINAL: this.options_.terminal || DEFAULTS.terminal,
107
+ DS_MERCHANT_ORDER: sessionData.orderId,
108
+ DS_MERCHANT_AMOUNT: sessionData.amount,
109
+ DS_MERCHANT_CURRENCY: sessionData.currency,
110
+ DS_MERCHANT_TRANSACTIONTYPE: "2",
111
+ };
112
+ const response = await this.redsysApi.restIniciaPeticion(params);
113
+ if (response.Ds_Response === "0000" ||
114
+ String(response.Ds_Response).startsWith("00")) {
115
+ sessionData.authCode = response.Ds_AuthorisationCode;
116
+ this.logger_.info("[REDSYS] Capture successful for order: " + sessionData.orderId);
117
+ }
118
+ else {
119
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.PAYMENT_AUTHORIZATION_ERROR, (0, errors_1.getErrorMessage)(response.Ds_Response));
120
+ }
121
+ return {
122
+ data: sessionData,
123
+ };
124
+ }
125
+ // ---------- Cancel ----------
126
+ async cancelPayment(input) {
127
+ const sessionData = input.data;
128
+ if (!sessionData?.orderId) {
129
+ return { data: input.data };
130
+ }
131
+ if (sessionData.status === "cancelled") {
132
+ return { data: input.data };
133
+ }
134
+ const params = {
135
+ DS_MERCHANT_MERCHANTCODE: this.options_.merchantCode,
136
+ DS_MERCHANT_TERMINAL: this.options_.terminal || DEFAULTS.terminal,
137
+ DS_MERCHANT_ORDER: sessionData.orderId,
138
+ DS_MERCHANT_AMOUNT: sessionData.amount,
139
+ DS_MERCHANT_CURRENCY: sessionData.currency,
140
+ DS_MERCHANT_TRANSACTIONTYPE: "9",
141
+ };
142
+ const response = await this.redsysApi.restIniciaPeticion(params);
143
+ if (response.Ds_Response === "0000" ||
144
+ String(response.Ds_Response).startsWith("00")) {
145
+ sessionData.status = "cancelled";
146
+ this.logger_.info("[REDSYS] Payment cancelled for order: " + sessionData.orderId);
147
+ }
148
+ else {
149
+ this.logger_.warn("[REDSYS] Cancellation responded with code: " +
150
+ response.Ds_Response);
151
+ }
152
+ return {
153
+ data: sessionData,
154
+ };
155
+ }
156
+ // ---------- Refund ----------
157
+ async refundPayment(input) {
158
+ const sessionData = input.data;
159
+ if (!sessionData?.orderId) {
160
+ return { data: input.data };
161
+ }
162
+ const refundAmount = this.assertPositiveAmount(input.amount);
163
+ const currencyToAlpha = {
164
+ "978": "eur",
165
+ "840": "usd",
166
+ "826": "gbp",
167
+ "392": "jpy",
168
+ };
169
+ const currencyCode = currencyToAlpha[sessionData.currency] || "eur";
170
+ const amountStr = String((0, amount_1.getSmallestUnit)(refundAmount, currencyCode));
171
+ const params = {
172
+ DS_MERCHANT_MERCHANTCODE: this.options_.merchantCode,
173
+ DS_MERCHANT_TERMINAL: this.options_.terminal || DEFAULTS.terminal,
174
+ DS_MERCHANT_ORDER: sessionData.orderId,
175
+ DS_MERCHANT_AMOUNT: amountStr,
176
+ DS_MERCHANT_CURRENCY: sessionData.currency,
177
+ DS_MERCHANT_TRANSACTIONTYPE: "3",
178
+ };
179
+ const response = await this.redsysApi.restIniciaPeticion(params);
180
+ const code = String(response.Ds_Response);
181
+ if (code === "0000" ||
182
+ code.startsWith("00") ||
183
+ code === "0900" ||
184
+ code === "900") {
185
+ sessionData.status = "refunded";
186
+ this.logger_.info("[REDSYS] Refund processed for order: " +
187
+ sessionData.orderId +
188
+ " Amount: " +
189
+ amountStr);
190
+ }
191
+ else {
192
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.PAYMENT_AUTHORIZATION_ERROR, (0, errors_1.getErrorMessage)(response.Ds_Response));
193
+ }
194
+ return {
195
+ data: sessionData,
196
+ };
197
+ }
198
+ // ---------- Status ----------
199
+ async getPaymentStatus(input) {
200
+ const sessionData = input.data;
201
+ if (!sessionData?.status) {
202
+ return { status: utils_1.PaymentSessionStatus.PENDING };
203
+ }
204
+ switch (sessionData.status) {
205
+ case "authorized":
206
+ return { status: utils_1.PaymentSessionStatus.AUTHORIZED };
207
+ case "refunded":
208
+ return { status: utils_1.PaymentSessionStatus.CAPTURED };
209
+ case "cancelled":
210
+ return { status: utils_1.PaymentSessionStatus.CANCELED };
211
+ case "error":
212
+ return { status: utils_1.PaymentSessionStatus.ERROR };
213
+ default:
214
+ return { status: utils_1.PaymentSessionStatus.PENDING };
215
+ }
216
+ }
217
+ // ---------- Retrieve ----------
218
+ async retrievePayment(input) {
219
+ return { data: input.data };
220
+ }
221
+ // ---------- Update ----------
222
+ async updatePayment(input) {
223
+ const sessionData = input.data;
224
+ const orderId = sessionData?.orderId || (0, order_id_1.generateOrderId)();
225
+ const sessionId = "redsys_" + orderId;
226
+ const amount = this.assertPositiveAmount(input.amount);
227
+ const amountStr = String((0, amount_1.getSmallestUnit)(amount, input.currency_code));
228
+ const currencyNum = (0, currency_1.getCurrencyNum)(input.currency_code);
229
+ const transactionType = this.options_.transactionType || DEFAULTS.transactionType;
230
+ const merchantParams = {
231
+ DS_MERCHANT_MERCHANTCODE: this.options_.merchantCode,
232
+ DS_MERCHANT_TERMINAL: this.options_.terminal || DEFAULTS.terminal,
233
+ DS_MERCHANT_ORDER: orderId,
234
+ DS_MERCHANT_AMOUNT: amountStr,
235
+ DS_MERCHANT_CURRENCY: currencyNum,
236
+ DS_MERCHANT_TRANSACTIONTYPE: transactionType,
237
+ DS_MERCHANT_CONSUMERLANGUAGE: "1",
238
+ };
239
+ if (this.options_.notificationUrl) {
240
+ merchantParams.DS_MERCHANT_MERCHANTURL = this.options_.notificationUrl;
241
+ }
242
+ const separator = (url) => url.includes("?") ? "&" : "?";
243
+ if (this.options_.successUrl) {
244
+ merchantParams.DS_MERCHANT_URLOK =
245
+ this.options_.successUrl + separator(this.options_.successUrl) + "orderId=" + orderId;
246
+ }
247
+ if (this.options_.errorUrl) {
248
+ merchantParams.DS_MERCHANT_URLKO =
249
+ this.options_.errorUrl + separator(this.options_.errorUrl) + "orderId=" + orderId;
250
+ }
251
+ merchantParams.DS_MERCHANT_MERCHANTDATA = sessionId + "|" + orderId;
252
+ const form = await this.redsysApi.createRedirectForm(merchantParams);
253
+ const newData = {
254
+ orderId,
255
+ amount: amountStr,
256
+ currency: currencyNum,
257
+ status: "pending",
258
+ transactionType,
259
+ merchantParams: form.body.Ds_MerchantParameters,
260
+ signature: form.body.Ds_Signature,
261
+ signatureVersion: form.body.Ds_SignatureVersion,
262
+ formUrl: form.url,
263
+ };
264
+ return {
265
+ data: newData,
266
+ };
267
+ }
268
+ // ---------- Delete ----------
269
+ async deletePayment(input) {
270
+ return {};
271
+ }
272
+ // ---------- Webhook ----------
273
+ async getWebhookActionAndData(payload) {
274
+ try {
275
+ const notification = this.redsysApi.processRestNotification(payload.data);
276
+ if (!notification) {
277
+ this.logger_.warn("[REDSYS] Webhook: invalid notification data");
278
+ return { action: utils_1.PaymentActions.NOT_SUPPORTED };
279
+ }
280
+ const dsResponse = String(notification.Ds_Response);
281
+ if (dsResponse === "0000" || dsResponse.startsWith("00")) {
282
+ this.logger_.info("[REDSYS] Webhook: payment authorized for order: " +
283
+ notification.Ds_Order);
284
+ let sessionId;
285
+ let orderId = notification.Ds_Order;
286
+ try {
287
+ const merchantData = notification.Ds_MerchantData;
288
+ if (merchantData) {
289
+ const parts = merchantData.split("|");
290
+ if (parts.length >= 3) {
291
+ orderId = parts[2];
292
+ sessionId = parts[1];
293
+ }
294
+ }
295
+ }
296
+ catch {
297
+ // MerchantData parsing is best-effort
298
+ }
299
+ return {
300
+ action: utils_1.PaymentActions.SUCCESSFUL,
301
+ data: {
302
+ session_id: sessionId || "redsys_" + orderId,
303
+ amount: notification.Ds_Amount || 0,
304
+ },
305
+ };
306
+ }
307
+ this.logger_.warn("[REDSYS] Webhook: payment not authorized. Order: " +
308
+ notification.Ds_Order +
309
+ " Response: " +
310
+ dsResponse);
311
+ return {
312
+ action: utils_1.PaymentActions.FAILED,
313
+ };
314
+ }
315
+ catch (error) {
316
+ this.logger_.error("[REDSYS] Webhook error: " + error.message);
317
+ return { action: utils_1.PaymentActions.NOT_SUPPORTED };
318
+ }
319
+ }
320
+ // ---------- Helpers ----------
321
+ assertPositiveAmount(amount) {
322
+ const n = parseFloat(String(amount));
323
+ if (!Number.isFinite(n) || n <= 0) {
324
+ throw new utils_1.MedusaError(utils_1.MedusaError.Types.INVALID_DATA, "amount must be a positive finite number, got \"" + amount + "\"");
325
+ }
326
+ return n;
327
+ }
328
+ }
329
+ RedsysProviderService.identifier = "redsys";
330
+ exports.default = RedsysProviderService;
331
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9wcm92aWRlcnMvcmVkc3lzL3NlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFBQSxxREFLa0M7QUFDbEMsNkNBQTRFO0FBOEI1RSwrQ0FBb0Q7QUFDcEQsbURBQXFEO0FBQ3JELG1EQUFzRDtBQUN0RCwrQ0FBb0Q7QUFNcEQsTUFBTSxRQUFRLEdBQUc7SUFDZixRQUFRLEVBQUUsS0FBSztJQUNmLGVBQWUsRUFBRSxHQUFHO0NBQ1osQ0FBQTtBQUVWLE1BQU0scUJBQXNCLFNBQVEsK0JBQXNDO0lBT3hFLE1BQU0sQ0FBQyxlQUFlLENBQUMsT0FBZ0M7UUFDckQsSUFBSSxDQUFDLE9BQU8sQ0FBQyxTQUFTLElBQUksT0FBTyxPQUFPLENBQUMsU0FBUyxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQ2hFLE1BQU0sSUFBSSxtQkFBVyxDQUNuQixtQkFBVyxDQUFDLEtBQUssQ0FBQyxZQUFZLEVBQzlCLG1EQUFtRCxDQUNwRCxDQUFBO1FBQ0gsQ0FBQztRQUNELElBQUksQ0FBQyxPQUFPLENBQUMsWUFBWSxJQUFJLE9BQU8sT0FBTyxDQUFDLFlBQVksS0FBSyxRQUFRLEVBQUUsQ0FBQztZQUN0RSxNQUFNLElBQUksbUJBQVcsQ0FDbkIsbUJBQVcsQ0FBQyxLQUFLLENBQUMsWUFBWSxFQUM5QixzREFBc0QsQ0FDdkQsQ0FBQTtRQUNILENBQUM7SUFDSCxDQUFDO0lBRUQsWUFBWSxTQUErQixFQUFFLE9BQXNCO1FBQ2pFLEtBQUssQ0FBQyxTQUFTLEVBQUUsT0FBTyxDQUFDLENBQUE7UUFDekIsSUFBSSxDQUFDLE9BQU8sR0FBRyxTQUFTLENBQUMsTUFBTSxDQUFBO1FBQy9CLElBQUksQ0FBQyxRQUFRLEdBQUcsT0FBTyxDQUFBO1FBRXZCLElBQUksQ0FBQyxTQUFTLEdBQUcsSUFBQSw2QkFBZSxFQUFDO1lBQy9CLFNBQVMsRUFBRSxPQUFPLENBQUMs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
@@ -0,0 +1,59 @@
1
+ "use strict";
2
+ /**
3
+ * REDSYS Payment Plugin Types
4
+ */
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.RedsysResponseCodes = exports.RedsysCurrencyCodes = exports.RedsysTransactionTypes = void 0;
7
+ /**
8
+ * REDSYS Transaction Types
9
+ */
10
+ exports.RedsysTransactionTypes = {
11
+ PAYMENT: "0",
12
+ PREAUTHORIZATION: "1",
13
+ CONFIRMATION: "2",
14
+ REFUND: "3",
15
+ CANCELLATION: "9",
16
+ };
17
+ /**
18
+ * REDSYS Currency Codes (ISO 4217 numeric)
19
+ */
20
+ exports.RedsysCurrencyCodes = {
21
+ EUR: "978",
22
+ USD: "840",
23
+ GBP: "826",
24
+ JPY: "392",
25
+ MXN: "484",
26
+ ARS: "032",
27
+ CLP: "152",
28
+ COP: "170",
29
+ BRL: "986",
30
+ CHF: "756",
31
+ DKK: "208",
32
+ NOK: "578",
33
+ SEK: "752",
34
+ PLN: "985",
35
+ CZK: "203",
36
+ HUF: "348",
37
+ RON: "946",
38
+ BGN: "975",
39
+ HRK: "191",
40
+ ISK: "352",
41
+ TRY: "949",
42
+ AUD: "036",
43
+ CAD: "124",
44
+ CNY: "156",
45
+ INR: "356",
46
+ KRW: "410",
47
+ RUB: "643",
48
+ ZAR: "710",
49
+ };
50
+ /**
51
+ * REDSYS Response Codes
52
+ */
53
+ exports.RedsysResponseCodes = {
54
+ AUTHORIZED: "0000",
55
+ AUTHORIZED_BELOW_100: "00",
56
+ REFUND_APPROVED: "0900",
57
+ REFUND_APPROVED_BELOW_900: "900",
58
+ };
59
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOztHQUVHOzs7QUF5Qkg7O0dBRUc7QUFDVSxRQUFBLHNCQUFzQixHQUFHO0lBQ3BDLE9BQU8sRUFBRSxHQUFHO0lBQ1osZ0JBQWdCLEVBQUUsR0FBRztJQUNyQixZQUFZLEVBQUUsR0FBRztJQUNqQixNQUFNLEVBQUUsR0FBRztJQUNYLFlBQVksRUFBRSxHQUFHO0NBQ1QsQ0FBQTtBQUVWOztHQUVHO0FBQ1UsUUFBQSxtQkFBbUIsR0FBMkI7SUFDekQsR0FBRyxFQUFFLEtBQUs7SUFDVixHQUFHLEVBQUUsS0FBSztJQUNWLEdBQUcsRUFBRSxLQUFLO0lBQ1YsR0FBRyxFQUFFLEtBQUs7SUFDVixHQUFHLEVBQUUsS0FBSztJQUNWLEdBQUcsRUFBRSxLQUFLO0lBQ1YsR0FBRyxFQUFFLEtBQUs7SUFDVixHQUFHLEVBQUUsS0FBSztJQUNWLEdBQUcsRUFBRSxLQUFLO0lBQ1YsR0FBRyxFQUFFLEtBQUs7SUFDVixHQUFHLEVBQUUsS0FBSztJQUNWLEdBQUcsRUFBRSxLQUFLO0lBQ1YsR0FBRyxFQUFFLEtBQUs7SUFDVixHQUFHLEVBQUUsS0FBSztJQUNWLEdBQUcsRUFBRSxLQUFLO0lBQ1YsR0FBRyxFQUFFLEtBQUs7SUFDVixHQUFHLEVBQUUsS0FBSztJQUNWLEdBQUcsRUFBRSxLQUFLO0lBQ1YsR0FBRyxFQUFFLEtBQUs7SUFDVixHQUFHLEVBQUUsS0FBSztJQUNWLEdBQUcsRUFBRSxLQUFLO0lBQ1YsR0FBRyxFQUFFLEtBQUs7SUFDVixHQUFHLEVBQUUsS0FBSztJQUNWLEdBQUcsRUFBRSxLQUFLO0lBQ1YsR0FBRyxFQUFFLEtBQUs7SUFDVixHQUFHLEVBQUUsS0FBSztJQUNWLEdBQUcsRUFBRSxLQUFLO0lBQ1YsR0FBRyxFQUFFLEtBQUs7Q0FDWCxDQUFBO0FBRUQ7O0dBRUc7QUFDVSxRQUFBLG1CQUFtQixHQUFHO0lBQ2pDLFVBQVUsRUFBRSxNQUFNO0lBQ2xCLG9CQUFvQixFQUFFLElBQUk7SUFDMUIsZUFBZSxFQUFFLE1BQU07SUFDdkIseUJBQXlCLEVBQUUsS0FBSztDQUN4QixDQUFBIn0=
@@ -0,0 +1,18 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.getSmallestUnit = getSmallestUnit;
4
+ /**
5
+ * Convert amount to smallest currency unit (cents for most currencies)
6
+ * Redsys expects amounts in the smallest unit (e.g., EUR cents)
7
+ */
8
+ function getSmallestUnit(amount, currency) {
9
+ const normalizedCurrency = currency.toLowerCase();
10
+ // Currencies without decimals (0 decimal places)
11
+ const zeroDecimal = ["jpy", "krw", "vnd", "isk", "clp"];
12
+ if (zeroDecimal.includes(normalizedCurrency)) {
13
+ return Math.round(amount);
14
+ }
15
+ // Standard currencies with 2 decimal places
16
+ return Math.round(amount * 100);
17
+ }
18
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYW1vdW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3V0aWxzL2Ftb3VudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUlBLDBDQVdDO0FBZkQ7OztHQUdHO0FBQ0gsU0FBZ0IsZUFBZSxDQUFDLE1BQWMsRUFBRSxRQUFnQjtJQUM5RCxNQUFNLGtCQUFrQixHQUFHLFFBQVEsQ0FBQyxXQUFXLEVBQUUsQ0FBQTtJQUVqRCxpREFBaUQ7SUFDakQsTUFBTSxXQUFXLEdBQUcsQ0FBQyxLQUFLLEVBQUUsS0FBSyxFQUFFLEtBQUssRUFBRSxLQUFLLEVBQUUsS0FBSyxDQUFDLENBQUE7SUFDdkQsSUFBSSxXQUFXLENBQUMsUUFBUSxDQUFDLGtCQUFrQixDQUFDLEVBQUUsQ0FBQztRQUM3QyxPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsTUFBTSxDQUFDLENBQUE7SUFDM0IsQ0FBQztJQUVELDRDQUE0QztJQUM1QyxPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsTUFBTSxHQUFHLEdBQUcsQ0FBQyxDQUFBO0FBQ2pDLENBQUMifQ==
@@ -0,0 +1,23 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.getCurrencyNum = getCurrencyNum;
4
+ exports.getCurrencyCode = getCurrencyCode;
5
+ const types_1 = require("../types");
6
+ /**
7
+ * Convert ISO 4217 currency code (e.g. "EUR") to Redsys numeric code (e.g. "978")
8
+ * Falls back to EUR ("978") for unknown currencies.
9
+ */
10
+ function getCurrencyNum(currencyCode) {
11
+ const upper = currencyCode?.toUpperCase() || "EUR";
12
+ return types_1.RedsysCurrencyCodes[upper] || "978";
13
+ }
14
+ /**
15
+ * Convert Redsys numeric currency code (e.g. "978") to ISO 4217 code (e.g. "EUR")
16
+ * Falls back to "eur" for unknown codes.
17
+ */
18
+ function getCurrencyCode(numericCode) {
19
+ const entries = Object.entries(types_1.RedsysCurrencyCodes);
20
+ const found = entries.find(([_, value]) => value === numericCode);
21
+ return found ? found[0].toLowerCase() : "eur";
22
+ }
23
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY3VycmVuY3kuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvdXRpbHMvY3VycmVuY3kudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFNQSx3Q0FHQztBQU1ELDBDQUlDO0FBbkJELG9DQUE4QztBQUU5Qzs7O0dBR0c7QUFDSCxTQUFnQixjQUFjLENBQUMsWUFBb0I7SUFDakQsTUFBTSxLQUFLLEdBQUcsWUFBWSxFQUFFLFdBQVcsRUFBRSxJQUFJLEtBQUssQ0FBQTtJQUNsRCxPQUFPLDJCQUFtQixDQUFDLEtBQUssQ0FBQyxJQUFJLEtBQUssQ0FBQTtBQUM1QyxDQUFDO0FBRUQ7OztHQUdHO0FBQ0gsU0FBZ0IsZUFBZSxDQUFDLFdBQW1CO0lBQ2pELE1BQU0sT0FBTyxHQUFHLE1BQU0sQ0FBQyxPQUFPLENBQUMsMkJBQW1CLENBQXVCLENBQUE7SUFDekUsTUFBTSxLQUFLLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUFFLEtBQUssQ0FBQyxFQUFFLEVBQUUsQ0FBQyxLQUFLLEtBQUssV0FBVyxDQUFDLENBQUE7SUFDakUsT0FBTyxLQUFLLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxXQUFXLEVBQUUsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFBO0FBQy9DLENBQUMifQ==
@@ -0,0 +1,49 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.getErrorMessage = getErrorMessage;
4
+ /**
5
+ * Human-readable Spanish error messages for REDSYS response codes.
6
+ */
7
+ function getErrorMessage(code) {
8
+ const codeStr = String(code);
9
+ const errorMessages = {
10
+ "101": "Tarjeta caducada o con límite excedido",
11
+ "102": "Tarjeta en sospecha de fraude",
12
+ "104": "Operación no permitida para esta tarjeta",
13
+ "106": "Tarjeta caducada",
14
+ "107": "Operación no permitida para este comercio",
15
+ "109": "Comercio no operativo temporalmente",
16
+ "110": "Importe excede el límite permitido",
17
+ "114": "Tipo de operación no permitida",
18
+ "116": "Saldo insuficiente",
19
+ "118": "Tarjeta no registrada",
20
+ "129": "Código de seguridad (CVV2) incorrecto",
21
+ "180": "Tarjeta no válida",
22
+ "190": "Denegación sin especificar",
23
+ "191": "Fecha de caducidad incorrecta",
24
+ "195": "Requiere autenticación SCA",
25
+ "202": "Tarjeta en sospecha de fraude con denegación",
26
+ "904": "Comercio no registrado",
27
+ "909": "Error de sistema",
28
+ "913": "Pedido repetido",
29
+ "944": "Sesión incorrecta",
30
+ "950": "Operación de devolución no permitida",
31
+ "9064": "Número de tarjeta incorrecto",
32
+ "9078": "Tipo de operación no permitida",
33
+ "9093": "Tarjeta no existente",
34
+ "9094": "Servicio no disponible para esta tarjeta",
35
+ "9104": "Comercio no operativo",
36
+ "9218": "Operación no permitida",
37
+ "9253": "Tarjeta bloqueada",
38
+ "9256": "Tarjeta no permite operaciones de preautorización",
39
+ "9257": "Tarjeta no permite operaciones de devolución",
40
+ "9261": "Límite de reintentos de pago superado",
41
+ "9912": "Emisor no disponible",
42
+ "9915": "Cancelación automática por timeout",
43
+ "9928": "Anulación de autorización no permitida",
44
+ "9998": "Operación no permitida (AVS)",
45
+ "9999": "Autenticación requerida",
46
+ };
47
+ return errorMessages[codeStr] || `Transacción denegada (código: ${codeStr})`;
48
+ }
49
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3JzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3V0aWxzL2Vycm9ycy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUdBLDBDQXlDQztBQTVDRDs7R0FFRztBQUNILFNBQWdCLGVBQWUsQ0FBQyxJQUFxQjtJQUNuRCxNQUFNLE9BQU8sR0FBRyxNQUFNLENBQUMsSUFBSSxDQUFDLENBQUE7SUFDNUIsTUFBTSxhQUFhLEdBQTJCO1FBQzVDLEtBQUssRUFBRSx3Q0FBd0M7UUFDL0MsS0FBSyxFQUFFLCtCQUErQjtRQUN0QyxLQUFLLEVBQUUsMENBQTBDO1FBQ2pELEtBQUssRUFBRSxrQkFBa0I7UUFDekIsS0FBSyxFQUFFLDJDQUEyQztRQUNsRCxLQUFLLEVBQUUscUNBQXFDO1FBQzVDLEtBQUssRUFBRSxvQ0FBb0M7UUFDM0MsS0FBSyxFQUFFLGdDQUFnQztRQUN2QyxLQUFLLEVBQUUsb0JBQW9CO1FBQzNCLEtBQUssRUFBRSx1QkFBdUI7UUFDOUIsS0FBSyxFQUFFLHVDQUF1QztRQUM5QyxLQUFLLEVBQUUsbUJBQW1CO1FBQzFCLEtBQUssRUFBRSw0QkFBNEI7UUFDbkMsS0FBSyxFQUFFLCtCQUErQjtRQUN0QyxLQUFLLEVBQUUsNEJBQTRCO1FBQ25DLEtBQUssRUFBRSw4Q0FBOEM7UUFDckQsS0FBSyxFQUFFLHdCQUF3QjtRQUMvQixLQUFLLEVBQUUsa0JBQWtCO1FBQ3pCLEtBQUssRUFBRSxpQkFBaUI7UUFDeEIsS0FBSyxFQUFFLG1CQUFtQjtRQUMxQixLQUFLLEVBQUUsc0NBQXNDO1FBQzdDLE1BQU0sRUFBRSw4QkFBOEI7UUFDdEMsTUFBTSxFQUFFLGdDQUFnQztRQUN4QyxNQUFNLEVBQUUsc0JBQXNCO1FBQzlCLE1BQU0sRUFBRSwwQ0FBMEM7UUFDbEQsTUFBTSxFQUFFLHVCQUF1QjtRQUMvQixNQUFNLEVBQUUsd0JBQXdCO1FBQ2hDLE1BQU0sRUFBRSxtQkFBbUI7UUFDM0IsTUFBTSxFQUFFLG1EQUFtRDtRQUMzRCxNQUFNLEVBQUUsOENBQThDO1FBQ3RELE1BQU0sRUFBRSx1Q0FBdUM7UUFDL0MsTUFBTSxFQUFFLHNCQUFzQjtRQUM5QixNQUFNLEVBQUUsb0NBQW9DO1FBQzVDLE1BQU0sRUFBRSx3Q0FBd0M7UUFDaEQsTUFBTSxFQUFFLDhCQUE4QjtRQUN0QyxNQUFNLEVBQUUseUJBQXlCO0tBQ2xDLENBQUE7SUFDRCxPQUFPLGFBQWEsQ0FBQyxPQUFPLENBQUMsSUFBSSxpQ0FBaUMsT0FBTyxHQUFHLENBQUE7QUFDOUUsQ0FBQyJ9
@@ -0,0 +1,19 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.generateOrderId = generateOrderId;
4
+ /**
5
+ * Generate a unique order ID for Redsys.
6
+ * Redsys requires:
7
+ * - First 4 characters must be digits
8
+ * - Max 12 characters
9
+ * - Alphanumeric (digits + uppercase letters)
10
+ */
11
+ function generateOrderId() {
12
+ const prefix = String(Math.floor(Math.random() * 9000 + 1000));
13
+ const suffix = Math.random()
14
+ .toString(36)
15
+ .substring(2, 10)
16
+ .toUpperCase();
17
+ return (prefix + suffix).substring(0, 12);
18
+ }
19
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib3JkZXItaWQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvdXRpbHMvb3JkZXItaWQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFPQSwwQ0FPQztBQWREOzs7Ozs7R0FNRztBQUNILFNBQWdCLGVBQWU7SUFDN0IsTUFBTSxNQUFNLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLE1BQU0sRUFBRSxHQUFHLElBQUksR0FBRyxJQUFJLENBQUMsQ0FBQyxDQUFBO0lBQzlELE1BQU0sTUFBTSxHQUFHLElBQUksQ0FBQyxNQUFNLEVBQUU7U0FDekIsUUFBUSxDQUFDLEVBQUUsQ0FBQztTQUNaLFNBQVMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDO1NBQ2hCLFdBQVcsRUFBRSxDQUFBO0lBQ2hCLE9BQU8sQ0FBQyxNQUFNLEdBQUcsTUFBTSxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQTtBQUMzQyxDQUFDIn0=
package/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2025 My Company
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/README.md ADDED
@@ -0,0 +1,497 @@
1
+ # medusa-payment-redsys
2
+
3
+ Redsys / Sermepa TPV Virtual payment provider plugin for [MedusaJS v2](https://medusajs.com/).
4
+
5
+ This plugin enables payment processing through Redsys' hosted payment page (TPV Virtual) via redirect flow. Customers are redirected to the Redsys secure payment page to complete their transaction.
6
+
7
+ > **Production-proven**: This plugin is derived from a live production Medusa store processing real Redsys payments.
8
+
9
+ ## Features
10
+
11
+ - Redsys hosted payment page / TPV Virtual redirect flow
12
+ - Sandbox and production environments
13
+ - One-step payment (immediate capture) and two-step payment (pre-authorization + capture)
14
+ - Full and partial refunds via Redsys API
15
+ - Payment cancellation
16
+ - Webhook handling with HMAC-SHA256 signature verification
17
+ - Spanish error messages for Redsys response codes
18
+ - Zero PCI scope — card data is handled by Redsys' secure page
19
+
20
+ ## Prerequisites
21
+
22
+ - MedusaJS v2.13.0 or later
23
+ - Node.js v20 or later
24
+ - A [Redsys merchant account](https://comercios.redsys.es/) (or sandbox test credentials)
25
+ - `redsys-easy` v5.3.0+ (installed automatically as a dependency)
26
+
27
+ ## Installation
28
+
29
+ ```bash
30
+ npm install medusa-payment-redsys
31
+ # or
32
+ yarn add medusa-payment-redsys
33
+ # or
34
+ pnpm add medusa-payment-redsys
35
+ ```
36
+
37
+ ## Configuration
38
+
39
+ ### Environment Variables
40
+
41
+ Add the following to your `.env` file:
42
+
43
+ ```env
44
+ REDSYS_SECRET_KEY=sq7Hj....
45
+ REDSYS_MERCHANT_CODE=999008881
46
+ REDSYS_TERMINAL=001
47
+ REDSYS_ENVIRONMENT=sandbox
48
+ REDSYS_NOTIFICATION_URL=https://your-api.com/hooks/payment/redsys_redsys
49
+ REDSYS_SUCCESS_URL=https://your-store.com/checkout/redsys-callback
50
+ REDSYS_ERROR_URL=https://your-store.com/checkout/redsys-callback?error=1
51
+ ```
52
+
53
+ For sandbox testing, use the following test credentials from Redsys:
54
+
55
+ ```
56
+ Merchant Code: 999008881
57
+ Terminal: 001
58
+ Secret Key: sq7Hj.......
59
+ Environment: sandbox
60
+ ```
61
+
62
+ ### Medusa Configuration
63
+
64
+ In your `medusa-config.ts`:
65
+
66
+ ```ts
67
+ import { defineConfig } from "@medusajs/framework/config"
68
+
69
+ export default defineConfig({
70
+ modules: [
71
+ {
72
+ resolve: "@medusajs/medusa/payment",
73
+ options: {
74
+ providers: [
75
+ {
76
+ resolve: "medusa-payment-redsys/providers/redsys",
77
+ id: "redsys",
78
+ options: {
79
+ secretKey: process.env.REDSYS_SECRET_KEY,
80
+ merchantCode: process.env.REDSYS_MERCHANT_CODE,
81
+ terminal: process.env.REDSYS_TERMINAL || "001",
82
+ environment:
83
+ process.env.REDSYS_ENVIRONMENT || "sandbox",
84
+ notificationUrl:
85
+ process.env.REDSYS_NOTIFICATION_URL,
86
+ successUrl: process.env.REDSYS_SUCCESS_URL,
87
+ errorUrl: process.env.REDSYS_ERROR_URL,
88
+ transactionType: "0", // "0" = immediate capture, "1" = pre-authorization
89
+ },
90
+ },
91
+ ],
92
+ },
93
+ },
94
+ ],
95
+ })
96
+ ```
97
+
98
+ ### Enable in Region
99
+
100
+ Enable the Redsys provider in your Medusa admin panel under **Settings > Regions** and select **Redsys** as a payment provider.
101
+
102
+ The provider ID will be:
103
+
104
+ ```
105
+ pp_redsys_redsys
106
+ ```
107
+
108
+ ## Options
109
+
110
+ | Option | Type | Required | Default | Description |
111
+ |---|---|---|---|---|
112
+ | `secretKey` | string | Yes | — | Redsys HMAC-SHA256 secret key |
113
+ | `merchantCode` | string | Yes | — | Redsys merchant code (FUC) |
114
+ | `terminal` | string | No | `"001"` | Terminal number |
115
+ | `environment` | string | No | `"sandbox"` | `"sandbox"` or `"production"` |
116
+ | `notificationUrl` | string | No | — | Webhook URL for Redsys to POST transaction results |
117
+ | `successUrl` | string | No | — | URL to redirect after successful payment (URLOK) |
118
+ | `errorUrl` | string | No | — | URL to redirect after failed payment (URLKO) |
119
+ | `transactionType` | string | No | `"0"` | `"0"` = immediate capture, `"1"` = pre-authorization |
120
+
121
+ ## Payment Flow
122
+
123
+ 1. Customer selects **Redsys** as payment method
124
+ 2. `initiatePayment()` creates a signed redirect form with Redsys merchant parameters
125
+ 3. Customer clicks "Place Order" → storefront calls `cart.complete()` to create the order, stores a cookie mapping the Redsys internal order ID to the Medusa order ID, then auto-submits the redirect form to Redsys TPV
126
+ 4. Customer completes payment on the Redsys hosted payment page
127
+ 5. Redsys sends a **webhook notification** to `{backendUrl}/hooks/payment/redsys_redsys`
128
+ 6. `getWebhookActionAndData()` validates the HMAC-SHA256 signature and updates the payment status
129
+ 7. Redsys redirects the customer's browser to `successUrl` or `errorUrl` with the Redsys order ID as a query parameter
130
+ 8. Storefront callback page reads the cookie to resolve the Medusa order ID and redirects to the order confirmation page
131
+
132
+ ### Important: authorizePayment Behavior
133
+
134
+ This plugin's `authorizePayment` returns `AUTHORIZED` for sessions with status `"pending"` **and** `"authorized"`. This is intentional for the redirect flow: the real authorization happens on Redsys TPV and is confirmed via webhook. Without this, `cart.complete()` would fail with a 400 error because Medusa requires the payment session to be authorized before completing the cart.
135
+
136
+ ### ID Mapping (Redsys → Medusa)
137
+
138
+ The plugin generates a 12-character alphanumeric `orderId` (e.g. `97727XYIWRRF`) used as Redsys' merchant order reference. When the order is completed via `cart.complete()`, Medusa generates its own order ID (e.g. `order_01KR3B4X...`). These are **different IDs**.
139
+
140
+ The callback URL from Redsys only contains the Redsys order ID, not the Medusa order ID. To bridge this gap, the storefront stores a cookie `redsys_map_{redsysOrderId}={medusaOrderId}` before redirecting to the TPV. The callback page reads this cookie to resolve the real Medusa order and redirect to the confirmation page.
141
+
142
+ ## Storefront Integration
143
+
144
+ Redsys is a **redirect-based** payment method (no card input in your storefront — the customer enters card data on Redsys' secure TPV). You must adapt your Medusa Next.js storefront with the changes below.
145
+
146
+ ### 1. `src/lib/constants.tsx` — Register the payment method
147
+
148
+ Add Redsys to the payment info map and add a helper function:
149
+
150
+ ```tsx
151
+ // Inside paymentInfoMap, add:
152
+ pp_redsys_redsys: {
153
+ title: "Credit / Debit Card",
154
+ icon: <CreditCard />,
155
+ },
156
+
157
+ // Add helper function:
158
+ export const isRedsys = (providerId?: string) => {
159
+ return providerId?.startsWith("pp_redsys_")
160
+ }
161
+ ```
162
+
163
+ ### 2. `src/lib/data/cart.ts` — Add order completion without redirect
164
+
165
+ Add a `completeCartWithoutRedirect` function. The standard `placeOrder` does a `redirect()` (server-side), but Redsys needs to redirect the browser to the TPV instead. This function completes the cart, creates the order, but returns the result so the client can handle the TPV redirect:
166
+
167
+ ```ts
168
+ export async function completeCartWithoutRedirect(cartId?: string) {
169
+ const id = cartId || (await getCartId())
170
+
171
+ if (!id) {
172
+ throw new Error("No existing cart found when completing cart")
173
+ }
174
+
175
+ const headers = {
176
+ ...(await getAuthHeaders()),
177
+ }
178
+
179
+ const cartRes = await sdk.store.cart
180
+ .complete(id, {}, headers)
181
+ .then(async (cartRes) => {
182
+ const cartCacheTag = await getCacheTag("carts")
183
+ revalidateTag(cartCacheTag)
184
+ return cartRes
185
+ })
186
+ .catch(medusaError)
187
+
188
+ if (cartRes?.type === "order") {
189
+ const orderCacheTag = await getCacheTag("orders")
190
+ revalidateTag(orderCacheTag)
191
+ removeCartId()
192
+ }
193
+
194
+ return cartRes
195
+ }
196
+ ```
197
+
198
+ ### 3. `src/modules/checkout/components/payment-button/index.tsx` — Redsys payment button
199
+
200
+ Add a `RedsysPaymentButton` component that:
201
+ 1. Reads the payment session data (formUrl, merchantParams, signature) from the cart
202
+ 2. Calls `completeCartWithoutRedirect()` to create the order
203
+ 3. Stores a cookie mapping the Redsys internal order ID → Medusa order ID (see [ID Mapping](#id-mapping-redsys--medusa))
204
+ 4. Dynamically builds and auto-submits a `<form>` to Redsys' TPV
205
+
206
+ ```tsx
207
+ // Add import:
208
+ import { isManual, isRedsys, isStripeLike } from "@lib/constants"
209
+ import { completeCartWithoutRedirect, placeOrder } from "@lib/data/cart"
210
+
211
+ // Add case in PaymentButton's switch:
212
+ case isRedsys(paymentSession?.provider_id):
213
+ return (
214
+ <RedsysPaymentButton
215
+ notReady={notReady}
216
+ cart={cart}
217
+ data-testid={dataTestId}
218
+ />
219
+ )
220
+
221
+ // Add the component:
222
+ const RedsysPaymentButton = ({
223
+ cart,
224
+ notReady,
225
+ "data-testid": dataTestId,
226
+ }: {
227
+ cart: HttpTypes.StoreCart
228
+ notReady: boolean
229
+ "data-testid"?: string
230
+ }) => {
231
+ const [submitting, setSubmitting] = useState(false)
232
+ const [errorMessage, setErrorMessage] = useState<string | null>(null)
233
+
234
+ const handlePayment = async () => {
235
+ setSubmitting(true)
236
+
237
+ const paymentSession = cart.payment_collection?.payment_sessions?.find(
238
+ (s) => s.status === "pending" && isRedsys(s.provider_id)
239
+ )
240
+
241
+ const redsysData = paymentSession?.data as Record<string, string> | undefined
242
+
243
+ if (!redsysData?.formUrl || !redsysData?.merchantParams || !redsysData?.signature) {
244
+ setErrorMessage("No se pudieron obtener los datos de pago de Redsys")
245
+ setSubmitting(false)
246
+ return
247
+ }
248
+
249
+ const cartRes = await completeCartWithoutRedirect()
250
+ .catch((err) => {
251
+ setErrorMessage(err.message)
252
+ setSubmitting(false)
253
+ return null
254
+ })
255
+
256
+ if (!cartRes || cartRes.type !== "order") {
257
+ setErrorMessage(cartRes ? "Error al crear el pedido" : "")
258
+ setSubmitting(false)
259
+ return
260
+ }
261
+
262
+ // Store mapping between Redsys internal orderId and Medusa orderId
263
+ // The callback page uses this cookie to resolve the real order
264
+ const medusaOrderId = cartRes.order.id
265
+ const redsysOrderId = redsysData.orderId || ""
266
+ document.cookie = `redsys_map_${redsysOrderId}=${medusaOrderId}; path=/; max-age=600; SameSite=Lax`
267
+
268
+ const form = document.createElement("form")
269
+ form.method = "POST"
270
+ form.action = redsysData.formUrl
271
+
272
+ const fields: Record<string, string> = {
273
+ Ds_SignatureVersion: redsysData.signatureVersion,
274
+ Ds_MerchantParameters: redsysData.merchantParams,
275
+ Ds_Signature: redsysData.signature,
276
+ }
277
+
278
+ Object.entries(fields).forEach(([name, value]) => {
279
+ const input = document.createElement("input")
280
+ input.type = "hidden"
281
+ input.name = name
282
+ input.value = value
283
+ form.appendChild(input)
284
+ })
285
+
286
+ document.body.appendChild(form)
287
+ form.submit()
288
+ }
289
+
290
+ return (
291
+ <>
292
+ <Button
293
+ disabled={notReady || submitting}
294
+ isLoading={submitting}
295
+ onClick={handlePayment}
296
+ size="large"
297
+ data-testid={dataTestId}
298
+ >
299
+ Place order
300
+ </Button>
301
+ <ErrorMessage
302
+ error={errorMessage}
303
+ data-testid="redsys-payment-error-message"
304
+ />
305
+ </>
306
+ )
307
+ }
308
+ ```
309
+
310
+ ### 4. `src/app/checkout/redsys-callback/page.tsx` — Callback page (new file)
311
+
312
+ Create the page Redsys redirects to after payment. The `orderId` query param from Redsys contains the **internal Redsys order ID**, not the Medusa order ID. Use the cookie set in step 3 to resolve the real Medusa order and redirect to the confirmation page:
313
+
314
+ ```tsx
315
+ import { retrieveOrder } from "@lib/data/orders"
316
+ import { cookies } from "next/headers"
317
+ import { Metadata } from "next"
318
+ import { redirect } from "next/navigation"
319
+
320
+ export const metadata: Metadata = {
321
+ title: "Resultado del pago",
322
+ description: "Resultado de la operación con Redsys",
323
+ }
324
+
325
+ type Props = {
326
+ searchParams: Promise<{ [key: string]: string | undefined }>
327
+ }
328
+
329
+ export default async function RedsysCallbackPage(props: Props) {
330
+ const searchParams = await props.searchParams
331
+
332
+ const isError = searchParams.error === "1"
333
+ const redsysOrderId = searchParams.orderId
334
+
335
+ if (isError) {
336
+ return (
337
+ <div className="flex flex-col items-center justify-center min-h-[50vh] gap-4 p-8">
338
+ <h1 className="text-2xl font-bold text-red-600">Pago no completado</h1>
339
+ <p className="text-gray-600">
340
+ La operación no se ha completado correctamente.
341
+ </p>
342
+ <a href="/" className="mt-4 px-6 py-2 bg-blue-600 text-white rounded-md hover:bg-blue-700">
343
+ Volver a la tienda
344
+ </a>
345
+ </div>
346
+ )
347
+ }
348
+
349
+ if (redsysOrderId) {
350
+ // Resolve the Medusa order ID from the cookie set by RedsysPaymentButton
351
+ const cookieStore = await cookies()
352
+ const medusaOrderId = cookieStore.get(`redsys_map_${redsysOrderId}`)?.value
353
+ const orderId = medusaOrderId || redsysOrderId
354
+
355
+ try {
356
+ const order = await retrieveOrder(orderId)
357
+ if (order) {
358
+ const countryCode = order.shipping_address?.country_code?.toLowerCase() || "dk"
359
+ return redirect(`/${countryCode}/order/${orderId}/confirmed`)
360
+ }
361
+ } catch {
362
+ // Order not found, show success anyway
363
+ }
364
+ }
365
+
366
+ return (
367
+ <div className="flex flex-col items-center justify-center min-h-[50vh] gap-4 p-8">
368
+ <h1 className="text-2xl font-bold text-green-600">Pago procesado</h1>
369
+ <p className="text-gray-600">Tu pago ha sido procesado correctamente.</p>
370
+ <a href="/" className="mt-4 px-6 py-2 bg-blue-600 text-white rounded-md hover:bg-blue-700">
371
+ Volver a la tienda
372
+ </a>
373
+ </div>
374
+ )
375
+ }
376
+ ```
377
+
378
+ ### 5. `src/middleware.ts` — Bypass region redirect
379
+
380
+ If your storefront uses middleware to enforce region/country code prefixes in URLs (as the default Medusa Next.js storefront does), add a bypass so `/checkout/redsys-callback` is not redirected. Add this early in the `middleware` function:
381
+
382
+ ```ts
383
+ // Redsys callback URL — bypass region redirect
384
+ if (request.nextUrl.pathname.startsWith("/checkout/redsys-callback")) {
385
+ return NextResponse.next()
386
+ }
387
+ ```
388
+
389
+ ### 6. `medusa-config.ts` — CORS
390
+
391
+ Ensure your storefront domain is allowed in CORS:
392
+
393
+ ```ts
394
+ projectConfig: {
395
+ http: {
396
+ storeCors: "http://localhost:8000,https://your-store.com",
397
+ },
398
+ }
399
+ ```
400
+
401
+ ### Session Data Reference
402
+
403
+ The payment session `data` field returned by `initiatePayment`:
404
+
405
+ ```ts
406
+ {
407
+ orderId: "1234ABCD5678",
408
+ amount: "2550",
409
+ currency: "978",
410
+ status: "pending",
411
+ transactionType: "0",
412
+ merchantParams: "base64...", // Base64-encoded merchant parameters
413
+ signature: "hmac...", // HMAC-SHA256 signature
414
+ signatureVersion: "HMAC_SHA256_V1",
415
+ formUrl: "https://sis-t.redsys.es:25443/sis/realizarPago"
416
+ }
417
+ ```
418
+
419
+ These fields are used in step 3 to build the auto-submitting redirect form.
420
+
421
+ ### Webhook
422
+
423
+ Medusa automatically exposes a webhook endpoint for the Redsys provider at:
424
+
425
+ ```
426
+ /hooks/payment/redsys_redsys
427
+ ```
428
+
429
+ For local development with sandbox, you must expose your backend to the internet (e.g., via [ngrok](https://ngrok.com/)) so Redsys can reach the webhook. Set `notificationUrl` to the ngrok URL.
430
+
431
+ **Important**: Redsys sends the notification to `notificationUrl` but the signature verification and payment status update happens through the Medusa webhook handler — make sure `notificationUrl` points to the same endpoint or forward notifications accordingly.
432
+
433
+ ## Test Cards (Sandbox)
434
+
435
+ | Card Number | Brand | Behavior |
436
+ |---|---|---|
437
+ | 4548810000000003 | VISA | 3DS v2 approved |
438
+ | 5576441563045037 | Mastercard | 3DS v2 approved |
439
+ | 4548814479727229 | VISA | 3DS frictionless |
440
+ | 4548817212493017 | VISA | 3DS challenge |
441
+ | Any + CVV 999 | Any | Payment declined |
442
+
443
+ ## Transaction Types
444
+
445
+ | Code | Type | Description |
446
+ |---|---|---|
447
+ | `"0"` | Payment | Authorization + immediate capture (default) |
448
+ | `"1"` | Pre-authorization | Reserve funds only |
449
+ | `"2"` | Confirmation | Capture pre-authorized funds |
450
+ | `"3"` | Refund | Full or partial refund |
451
+ | `"9"` | Cancellation | Cancel/void a transaction |
452
+
453
+ ## Security
454
+
455
+ - **Never log PAN, CVV, or the secret key**. The provider strips sensitive fields from log output.
456
+ - **Always validate signatures server-side**. `getWebhookActionAndData()` uses `redsys-easy`'s `processRestNotification()` for HMAC-SHA256 verification.
457
+ - **Use HTTPS** for all communication with Redsys.
458
+ - **Do not trust client-side payment data**. The webhook with signature verification is the source of truth.
459
+ - The redirect flow keeps you out of PCI scope — card data is handled by Redsys' secure page.
460
+
461
+ ## Currency Support
462
+
463
+ The plugin includes built-in numeric currency codes for all major currencies. If your currency is not listed, it defaults to EUR (`978`). See `src/types.ts` for the full list.
464
+
465
+ ## Development
466
+
467
+ ```bash
468
+ # Install dependencies
469
+ npm install
470
+
471
+ # Build
472
+ npm run build
473
+
474
+ # Run tests
475
+ npm test
476
+
477
+ # Watch mode (for local plugin development)
478
+ npm run dev
479
+ ```
480
+
481
+ ### Local Testing with a Medusa Project
482
+
483
+ ```bash
484
+ # From your plugin directory
485
+ npm run dev
486
+
487
+ # In your Medusa project directory:
488
+ npx medusa plugin:add ../path-to/medusa-payment-redsys
489
+ ```
490
+
491
+ ## License
492
+
493
+ MIT — see [LICENSE](./LICENSE) file for details.
494
+
495
+ ## Support
496
+
497
+ For issues and questions, please open an issue on [GitHub](https://github.com/juansoler/medusa-payment-redsys/issues).
package/package.json ADDED
@@ -0,0 +1,64 @@
1
+ {
2
+ "name": "@jsm406/medusa-plugin-redsys",
3
+ "version": "1.0.3",
4
+ "description": "Redsys / Sermepa TPV Virtual payment provider plugin for MedusaJS v2",
5
+ "author": "Juan Soler Márquez",
6
+ "license": "MIT",
7
+ "repository": {
8
+ "type": "git",
9
+ "url": "git+https://github.com/juansoler/medusa-payment-redsys.git"
10
+ },
11
+ "homepage": "https://github.com/juansoler/medusa-payment-redsys",
12
+ "publishConfig": {
13
+ "access": "public"
14
+ },
15
+ "files": [
16
+ ".medusa/server"
17
+ ],
18
+ "exports": {
19
+ "./package.json": "./package.json",
20
+ "./providers/*": "./.medusa/server/src/providers/*/index.js",
21
+ "./*": "./.medusa/server/src/*.js"
22
+ },
23
+ "keywords": [
24
+ "medusa",
25
+ "medusa-v2",
26
+ "medusa-plugin",
27
+ "medusa-plugin-payment",
28
+ "medusa-plugin-integration",
29
+ "redsys",
30
+ "sermepa",
31
+ "tpv",
32
+ "tpv-virtual",
33
+ "payment",
34
+ "spain",
35
+ "ecommerce"
36
+ ],
37
+ "scripts": {
38
+ "build": "medusa plugin:build",
39
+ "dev": "medusa plugin:develop",
40
+ "test": "vitest run",
41
+ "test:watch": "vitest",
42
+ "prepublishOnly": "medusa plugin:build"
43
+ },
44
+ "dependencies": {
45
+ "redsys-easy": "^5.3.0"
46
+ },
47
+ "peerDependencies": {
48
+ "@medusajs/framework": "^2.13.0",
49
+ "@medusajs/medusa": "^2.13.0"
50
+ },
51
+ "devDependencies": {
52
+ "@medusajs/cli": "^2.13.0",
53
+ "@medusajs/framework": "^2.13.0",
54
+ "@medusajs/medusa": "^2.13.0",
55
+ "@swc/core": "^1.15.33",
56
+ "@types/node": "^20.0.0",
57
+ "ts-node": "^10.9.2",
58
+ "typescript": "^5.6.2",
59
+ "vitest": "^3.0.0"
60
+ },
61
+ "engines": {
62
+ "node": ">=20"
63
+ }
64
+ }