├── .gitignore
├── .jshintrc
├── CHANGELOG.md
├── Gruntfile.js
├── LICENSE-MIT
├── README.md
├── dist
├── rxp-js.js
└── rxp-js.min.js
├── examples
└── hpp
│ ├── helper.js
│ ├── json
│ └── process-a-payment.json
│ ├── process-a-payment-embedded-autoload-callback.html
│ ├── process-a-payment-embedded-autoload.html
│ ├── process-a-payment-embedded.html
│ ├── process-a-payment-lightbox-callback.html
│ ├── process-a-payment-lightbox.html
│ ├── proxy-request.php
│ ├── redirect-for-payment.html
│ └── response.php
├── lib
├── .jshintrc
├── rxp-hpp.js
└── rxp-remote.js
├── package.json
└── specs
├── functional
└── hpp
│ ├── embedded-positives_spec.js
│ ├── lightbox-positives_spec.js
│ └── redirect-positives_spec.js
├── helpers
└── hpp.js
├── intern.config.js
└── unit
├── rxp-hpp_spec.js
└── rxp-remote_spec.js
/.gitignore:
--------------------------------------------------------------------------------
1 | /node_modules/
2 | yarn.lock
3 | package-lock.json
4 | .DS_Store
5 | _SpecRunner.html
6 | .grunt/
7 |
--------------------------------------------------------------------------------
/.jshintrc:
--------------------------------------------------------------------------------
1 | {
2 | "curly": true,
3 | "eqeqeq": true,
4 | "immed": true,
5 | "latedef": true,
6 | "newcap": true,
7 | "noarg": true,
8 | "sub": true,
9 | "undef": true,
10 | "unused": true,
11 | "boss": true,
12 | "eqnull": true,
13 | "node": true
14 | }
15 |
--------------------------------------------------------------------------------
/CHANGELOG.md:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | # Changelog
6 |
7 | ## Latest Version - v1.5.5 (10/15/24)
8 | - Added a logger for the steps performed by the library
9 |
10 | ## v1.5.4 (07/02/24)
11 | #### Enhancements:
12 | - Allow digital wallets payments when the mode is set to embedded
13 |
14 | ## v1.5.3 (03/04/24)
15 | #### Enhancements:
16 | - Allow digital wallets payments when the mode is set to lightbox
17 |
18 | ## v1.5.2 (06/08/23)
19 | #### Enhancements:
20 | - Allow the communication from Unified Payments
21 | - Update for the code examples
22 | ---
23 |
--------------------------------------------------------------------------------
/Gruntfile.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | module.exports = function(grunt) {
4 |
5 | // Project configuration.
6 | grunt.initConfig({
7 | // Metadata.
8 | pkg: grunt.file.readJSON('package.json'),
9 | banner: '/*! <%= pkg.name %> - v<%= pkg.version %> - <%= grunt.template.today("yyyy-mm-dd") %>' +
10 | '\n * <%= pkg.description %>' +
11 | '<%= pkg.homepage ? "\\n * " + pkg.homepage : "" %>' +
12 | '\n * Licensed <%= _.map(pkg.licenses, "type").join(", ") %>' +
13 | '\n */\n',
14 | // Task configuration.
15 | concat: {
16 | options: {
17 | banner: '<%= banner %>',
18 | stripBanners: true
19 | },
20 | dist: {
21 | src: ['lib/*.js'],
22 | dest: 'dist/<%= pkg.name %>.js'
23 | },
24 | },
25 | uglify: {
26 | options: {
27 | banner: '<%= banner %>'
28 | },
29 | dist: {
30 | src: '<%= concat.dist.dest %>',
31 | dest: 'dist/<%= pkg.name %>.min.js'
32 | },
33 | },
34 | jshint: {
35 | options: {
36 | jshintrc: '.jshintrc'
37 | },
38 | gruntfile: {
39 | src: 'Gruntfile.js'
40 | },
41 | lib: {
42 | options: {
43 | jshintrc: 'lib/.jshintrc'
44 | },
45 | src: ['lib/**/*.js']
46 | }
47 | },
48 | watch: {
49 | gruntfile: {
50 | files: '<%= jshint.gruntfile.src %>',
51 | tasks: ['jshint:gruntfile']
52 | },
53 | lib: {
54 | files: '<%= jshint.lib.src %>',
55 | tasks: ['jshint:lib', 'jasmine']
56 | },
57 | specs: {
58 | files: 'specs/unit/*.js',
59 | tasks: ['jasmine']
60 | }
61 | },
62 | jasmine : {
63 | src : 'lib/*.js',
64 | options: {
65 | specs: 'specs/unit/*spec.js',
66 | helpers: 'specs/unit/*helper.js'
67 | }
68 | },
69 | intern: {
70 | client: {
71 | options: {
72 | config: 'specs/intern.config'
73 | }
74 | },
75 | runner: {
76 | options: {
77 | config: 'specs/intern.config',
78 | runType: 'runner',
79 | // leaveRemoteOpen: true
80 | }
81 | }
82 | },
83 | php: {
84 | test: {
85 | options: {
86 | port: '8989',
87 | silent: true,
88 | }
89 | }
90 | }
91 | });
92 |
93 | // These plugins provide necessary tasks.
94 | grunt.loadNpmTasks('grunt-contrib-concat');
95 | grunt.loadNpmTasks('grunt-contrib-uglify');
96 | grunt.loadNpmTasks('grunt-contrib-jshint');
97 | grunt.loadNpmTasks('grunt-contrib-watch');
98 | grunt.loadNpmTasks('grunt-contrib-jasmine');
99 | grunt.loadNpmTasks('intern');
100 | grunt.loadNpmTasks('grunt-php');
101 |
102 | grunt.registerTask('test:functional', ['php', 'intern:runner']);
103 | grunt.registerTask('test:unit', ['jasmine']);
104 | grunt.registerTask('test', ['test:unit', 'test:functional']);
105 |
106 | // Default task.
107 | grunt.registerTask('default', ['jshint', 'concat', 'uglify']);
108 |
109 | };
110 |
--------------------------------------------------------------------------------
/LICENSE-MIT:
--------------------------------------------------------------------------------
1 | The MIT License (MIT)
2 |
3 | Copyright (c) 2018 Pay and Shop Ltd t/a Global Payments
4 |
5 | Permission is hereby granted, free of charge, to any person
6 | obtaining a copy of this software and associated documentation
7 | files (the "Software"), to deal in the Software without
8 | restriction, including without limitation the rights to use,
9 | copy, modify, merge, publish, distribute, sublicense, and/or sell
10 | copies of the Software, and to permit persons to whom the
11 | Software is furnished to do so, subject to the following
12 | conditions:
13 |
14 | The above copyright notice and this permission notice shall be
15 | included in all copies or substantial portions of the Software.
16 |
17 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
18 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
19 | OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
20 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
21 | HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
22 | WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
23 | FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
24 | OTHER DEALINGS IN THE SOFTWARE.
25 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Realex JS Library
2 | You can sign up for a Realex account at https://developer.realexpayments.com
3 |
4 | ## Hosted Payment Page (HPP) JS Library
5 |
6 | ### Usage
7 | The Javascript required to initialise the library is below. This code must only be executed when the DOM is fully loaded. (default method: lightbox)
8 | ```javascript
9 | RealexHpp.init(payButtonId, merchantUrl, jsonFromServerSdk[, options]);
10 | ```
11 | * payButtonId - The ID of the button used to launch the lightbox. Set to "autoload" if you want to load without having to press a button
12 | * merchantUrl - could be one of 2 types:
13 | - string - The URL to which the JSON response from Realex will be posted.
14 | - function - the callback function
15 | * jsonFromServerSdk - The JSON output from the Realex HPP Server SDK.
16 | * options/events
17 | - onResize (iframe embed) Send resize iframe events so the parent frame can be adjusted
18 |
19 | ### Enable the logger
20 | The following code enables a logger that will emit an event on each step performed by the library. Should be used for debugging only.
21 | ```javascript
22 | RealexHpp.setConfigItem('enableLogging', true);
23 | window.addEventListener(RealexHpp.constants.logEventName, function(e) {
24 | console.log(e.detail);
25 | });
26 | ```
27 |
28 | ### Consuming the resulting POST
29 | Once the payment has completed the Realex JSON response will be posted within to the supplied merchantUrl. The name of the field containing the JSON response is hppResponse.
30 |
31 | If you prefer to handle response manually, provide your own callback function in "merchantUrl". The answer will be pre-parsed to an object ready to be used.
32 |
33 | ## Examples
34 |
35 | * [embedded iFrame](examples/hpp/process-a-payment-embedded.html)
36 | * [embedded iFrame autoload](examples/hpp/process-a-payment-embedded-autoload.html)
37 | * [embedded iFrame autoload, callback](examples/hpp/process-a-payment-embedded-autoload-callback.html)
38 | * [lightbox/modal](examples/hpp/process-a-payment-lightbox.html)
39 |
40 | ## Remote JS Library
41 |
42 | ### Validation functions
43 | * validateCardNumber - validates card number format and performs a Luhn check
44 | * validateCardHolderName - validates card holder name is made up from ISO/IEC 8859-1:1998 characters
45 | * validateCvn - validates non-Amex CVN
46 | * validateAmexCvn - validates Amex CVN
47 | * validateExpiryDateFormat - validates expiry date format
48 | * validateExpiryDateNotInPast - validates expiry date is not in past
49 |
50 | ### Usage
51 | ```javascript
52 | RealexRemote.validateCardNumber(cardNumber);
53 | RealexRemote.validateCardHolderName(cardHolderName);
54 | RealexRemote.validateCvn(cvn);
55 | RealexRemote.validateAmexCvn(amexCvn);
56 | RealexRemote.validateExpiryDateFormat(expiryDate);
57 | RealexRemote.validateExpiryDateNotInPast(expiryDate);
58 | ```
59 |
60 | ## License
61 | See the LICENSE file.
62 |
--------------------------------------------------------------------------------
/dist/rxp-js.js:
--------------------------------------------------------------------------------
1 | /*! rxp-js - v1.5.5 - 2024-10-15
2 | * The official Realex Payments JS Library
3 | * https://github.com/realexpayments/rxp-js
4 | * Licensed MIT
5 | */
6 | Element.prototype.remove = function() {
7 | this.parentElement.removeChild(this);
8 | };
9 | NodeList.prototype.remove = HTMLCollection.prototype.remove = function() {
10 | for(var i = this.length - 1; i >= 0; i--) {
11 | if(this[i] && this[i].parentElement) {
12 | this[i].parentElement.removeChild(this[i]);
13 | }
14 | }
15 | };
16 | var RealexHpp = (function () {
17 |
18 | 'use strict';
19 |
20 | var hppUrl = "https://pay.realexpayments.com/pay";
21 |
22 | var allowedHppUrls = [
23 | 'https://pay.realexpayments.com/pay',
24 | 'https://pay.sandbox.realexpayments.com/pay'
25 | ];
26 |
27 | var randomId = randomId || Math.random().toString(16).substr(2,8);
28 |
29 | var setHppUrl = function(url) {
30 | hppUrl = url;
31 | };
32 |
33 | var mobileXSLowerBound = 360;
34 | var setMobileXSLowerBound = function (lowerBound) {
35 | mobileXSLowerBound = lowerBound;
36 | };
37 |
38 | var config = {
39 | enableLogging: false
40 | };
41 | var setConfigItem = function(configItem, value) {
42 | if (!config.hasOwnProperty(configItem)) {
43 | return;
44 | }
45 | config[configItem] = value;
46 | };
47 | var constants = {
48 | logEventName: 'rxp-log'
49 | };
50 | var eventMessages = {
51 | form: {
52 | append: 'Form appended to the iframe',
53 | create: 'Hidden form created',
54 | submit: 'Form submitted'
55 | },
56 | iFrame: {
57 | create: 'iFrame created',
58 | find: 'iFrame found'
59 | },
60 | initialize: function(mode) {
61 | return mode + ' initialized';
62 | }
63 | }
64 | var logEvent = function(message, data = {}) {
65 | if (!config.enableLogging) {
66 | return;
67 | }
68 |
69 | var event = new CustomEvent(constants.logEventName, { detail: { message: message, data: data } });
70 | window.dispatchEvent(event);
71 | };
72 |
73 | var isWindowsMobileOs = /Windows Phone|IEMobile/.test(navigator.userAgent);
74 | var isAndroidOrIOs = /Android|iPad|iPhone|iPod/.test(navigator.userAgent);
75 | var isMobileXS = function () {
76 | return (((window.innerWidth > 0) ? window.innerWidth : screen.width) <= mobileXSLowerBound ? true : false) ||
77 | (((window.innerHeight > 0) ? window.innerHeight : screen.Height) <= mobileXSLowerBound ? true : false);
78 | };
79 |
80 | // Display IFrame on WIndows Phone OS mobile devices
81 | var isMobileIFrame = isWindowsMobileOs;
82 |
83 | // For IOs/Android and small screen devices always open in new tab/window
84 | var isMobileNewTab = function () {
85 | return !isWindowsMobileOs && (isAndroidOrIOs || isMobileXS());
86 | };
87 |
88 | var tabWindow;
89 |
90 | var redirectUrl;
91 |
92 | /**
93 | * Shared functionality across lightbox, embedded, and redirect display modes.
94 | */
95 | var internal = {
96 | evtMsg: [],
97 | /**
98 | * Adds a new window message event listener and tracks it for later removal
99 | *
100 | * @param {Function} evtMsgFct
101 | */
102 | addEvtMsgListener: function(evtMsgFct) {
103 | this.evtMsg.push({ fct: evtMsgFct, opt: false });
104 | if (window.addEventListener) {
105 | window.addEventListener("message", evtMsgFct, false);
106 | } else {
107 | window.attachEvent('message', evtMsgFct);
108 | }
109 | },
110 | /**
111 | * Removes a previously set window message event listener
112 | */
113 | removeOldEvtMsgListener: function () {
114 | if (this.evtMsg.length > 0) {
115 | var evt = this.evtMsg.pop();
116 | if (window.addEventListener) {
117 | window.removeEventListener("message", evt.fct, evt.opt);
118 | } else {
119 | window.detachEvent('message', evt.fct);
120 | }
121 | }
122 | },
123 | /**
124 | * Shimmed base64 encode/decode support
125 | */
126 | base64:{
127 | encode:function(input) {
128 | var keyStr = "ABCDEFGHIJKLMNOP" +
129 | "QRSTUVWXYZabcdef" +
130 | "ghijklmnopqrstuv" +
131 | "wxyz0123456789+/" +
132 | "=";
133 | input = escape(input);
134 | var output = "";
135 | var chr1, chr2, chr3 = "";
136 | var enc1, enc2, enc3, enc4 = "";
137 | var i = 0;
138 |
139 | do {
140 | chr1 = input.charCodeAt(i++);
141 | chr2 = input.charCodeAt(i++);
142 | chr3 = input.charCodeAt(i++);
143 |
144 | enc1 = chr1 >> 2;
145 | enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);
146 | enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);
147 | enc4 = chr3 & 63;
148 |
149 | if (isNaN(chr2)) {
150 | enc3 = enc4 = 64;
151 | } else if (isNaN(chr3)) {
152 | enc4 = 64;
153 | }
154 |
155 | output = output +
156 | keyStr.charAt(enc1) +
157 | keyStr.charAt(enc2) +
158 | keyStr.charAt(enc3) +
159 | keyStr.charAt(enc4);
160 | chr1 = chr2 = chr3 = "";
161 | enc1 = enc2 = enc3 = enc4 = "";
162 | } while (i < input.length);
163 |
164 | return output;
165 | },
166 | decode:function(input) {
167 | if(typeof input === 'undefined') {
168 | return input;
169 | }
170 | var keyStr = "ABCDEFGHIJKLMNOP" +
171 | "QRSTUVWXYZabcdef" +
172 | "ghijklmnopqrstuv" +
173 | "wxyz0123456789+/" +
174 | "=";
175 | var output = "";
176 | var chr1, chr2, chr3 = "";
177 | var enc1, enc2, enc3, enc4 = "";
178 | var i = 0;
179 |
180 | // remove all characters that are not A-Z, a-z, 0-9, +, /, or =
181 | var base64test = /[^A-Za-z0-9\+\/\=]/g;
182 | if (base64test.exec(input)) {
183 | throw new Error("There were invalid base64 characters in the input text.\n" +
184 | "Valid base64 characters are A-Z, a-z, 0-9, '+', '/',and '='\n" +
185 | "Expect errors in decoding.");
186 | }
187 | input = input.replace(/[^A-Za-z0-9\+\/\=]/g, "");
188 |
189 | do {
190 | enc1 = keyStr.indexOf(input.charAt(i++));
191 | enc2 = keyStr.indexOf(input.charAt(i++));
192 | enc3 = keyStr.indexOf(input.charAt(i++));
193 | enc4 = keyStr.indexOf(input.charAt(i++));
194 |
195 | chr1 = (enc1 << 2) | (enc2 >> 4);
196 | chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);
197 | chr3 = ((enc3 & 3) << 6) | enc4;
198 |
199 | output = output + String.fromCharCode(chr1);
200 |
201 | if (enc3 !== 64) {
202 | output = output + String.fromCharCode(chr2);
203 | }
204 | if (enc4 !== 64) {
205 | output = output + String.fromCharCode(chr3);
206 | }
207 |
208 | chr1 = chr2 = chr3 = "";
209 | enc1 = enc2 = enc3 = enc4 = "";
210 |
211 | } while (i < input.length);
212 |
213 | return unescape(output);
214 | }
215 | },
216 | /**
217 | * Converts an HPP message to a developer-friendly version.
218 | *
219 | * The decode process has two steps:
220 | *
221 | * 1. Attempt to parse the string as JSON. If this fails, an error response
222 | * is provided as we expect that the HPP has errored out to the cardholder
223 | * 2. Attempt to base64 decode the data to cover both HPP versions 1 and 2.
224 | *
225 | * @param {any} answer
226 | * @returns null if answer is not a string, otherwise the data from the HPP
227 | */
228 | decodeAnswer:function(answer){ //internal.decodeAnswer
229 |
230 | var _r;
231 |
232 | if (typeof answer !== "string") {
233 | return null;
234 | }
235 |
236 | try {
237 | _r=JSON.parse(answer);
238 | } catch (e) {
239 | _r = { error: true, message: answer };
240 | }
241 |
242 | try {
243 | for(var r in _r){
244 | if(_r[r]) {
245 | _r[r]=internal.base64.decode(_r[r]);
246 | }
247 | }
248 | } catch (e) { /** */ }
249 | return _r;
250 | },
251 | /**
252 | * Creates a new input of type `hidden`. Does not append to DOM.
253 | *
254 | * @param {string} name Name for the new input
255 | * @param {string} value Value for the new input
256 | * @returns the created input
257 | */
258 | createFormHiddenInput: function (name, value) {
259 | var el = document.createElement("input");
260 | el.setAttribute("type", "hidden");
261 | el.setAttribute("name", name);
262 | el.setAttribute("value", value);
263 | return el;
264 | },
265 |
266 | /**
267 | * Determines a mobile device's orientation for width calculation
268 | *
269 | * @returns true if in landscape
270 | */
271 | checkDevicesOrientation: function () {
272 | if (window.orientation === 90 || window.orientation === -90) {
273 | return true;
274 | } else {
275 | return false;
276 | }
277 | },
278 |
279 | /**
280 | * Creates a semi-transparent overlay with full width/height to serve as
281 | * a background for the lightbox modal
282 | *
283 | * @returns the created overlay
284 | */
285 | createOverlay: function () {
286 | var overlay = document.createElement("div");
287 | overlay.setAttribute("id", "rxp-overlay-" + randomId);
288 | overlay.style.position = "fixed";
289 | overlay.style.width = "100%";
290 | overlay.style.height = "100%";
291 | overlay.style.top = "0";
292 | overlay.style.left = "0";
293 | overlay.style.transition = "all 0.3s ease-in-out";
294 | overlay.style.zIndex = "100";
295 |
296 | if (isMobileIFrame) {
297 | overlay.style.position = "absolute !important";
298 | overlay.style.WebkitOverflowScrolling = "touch";
299 | overlay.style.overflowX = "hidden";
300 | overlay.style.overflowY = "scroll";
301 | }
302 |
303 | document.body.appendChild(overlay);
304 |
305 | setTimeout(function () {
306 | overlay.style.background = "rgba(0, 0, 0, 0.7)";
307 | }, 1);
308 |
309 | return overlay;
310 | },
311 |
312 | /**
313 | * Closes a lightbox modal and all associated elements
314 | *
315 | * @param {HTMLImageElement} closeButton
316 | * @param {HTMLIFrameElement} iFrame
317 | * @param {HTMLImageElement} spinner
318 | * @param {HTMLDivElement} overlayElement
319 | */
320 | closeModal: function (closeButton, iFrame, spinner, overlayElement) {
321 | if (closeButton && closeButton.parentNode) {
322 | closeButton.parentNode.removeChild(closeButton);
323 | }
324 |
325 | if (iFrame && iFrame.parentNode) {
326 | iFrame.parentNode.removeChild(iFrame);
327 | }
328 |
329 | if (spinner && spinner.parentNode) {
330 | spinner.parentNode.removeChild(spinner);
331 | }
332 |
333 | if (!overlayElement) {
334 | return;
335 | }
336 |
337 | overlayElement.className = "";
338 | setTimeout(function () {
339 | if (overlayElement.parentNode) {
340 | overlayElement.parentNode.removeChild(overlayElement);
341 | }
342 | }, 300);
343 | },
344 |
345 | /**
346 | * Creates a close button for the lightbox modal
347 | *
348 | * @returns the created element
349 | */
350 | createCloseButton: function (overlayElement) {
351 | if (document.getElementById("rxp-frame-close-" + randomId) !== null) {
352 | return;
353 | }
354 |
355 | var closeButton = document.createElement("img");
356 | closeButton.setAttribute("id","rxp-frame-close-" + randomId);
357 | closeButton.setAttribute("src", "");
358 | closeButton.setAttribute("style","transition: all 0.5s ease-in-out; opacity: 0; float: left; position: absolute; left: 50%; margin-left: 173px; z-index: 99999999; top: 30px;");
359 |
360 | setTimeout(function () {
361 | closeButton.style.opacity = "1";
362 | },500);
363 |
364 | if (isMobileIFrame) {
365 | closeButton.style.position = "absolute";
366 | closeButton.style.float = "right";
367 | closeButton.style.top = "20px";
368 | closeButton.style.left = "initial";
369 | closeButton.style.marginLeft = "0px";
370 | closeButton.style.right = "20px";
371 | }
372 |
373 | return closeButton;
374 | },
375 |
376 | /**
377 | * Creates a form and appends the HPP request data as hidden input elements to
378 | * POST to the defined HPP URL.
379 | *
380 | * The created form is not appended to the DOM and is not submitted at this time.
381 | *
382 | * @param {Document} doc
383 | * @param {object} token HPP request data
384 | * @param {bool} ignorePostMessage If true, the HPP will redirect to the defined
385 | * defined redirect URL. If false, the HPP will send a postMessage
386 | * to the parent window to be handled by this library.
387 | * @returns the created form
388 | */
389 | createForm: function (doc, token, ignorePostMessage) {
390 | var form = document.createElement("form");
391 | form.setAttribute("method", "POST");
392 | form.setAttribute("action", hppUrl);
393 |
394 | var versionSet = false;
395 |
396 | for (var key in token) {
397 | if (key === "HPP_VERSION"){
398 | versionSet = true;
399 | }
400 | form.appendChild(internal.createFormHiddenInput(key, token[key]));
401 | }
402 |
403 | if (versionSet === false){
404 | form.appendChild(internal.createFormHiddenInput("HPP_VERSION", "2"));
405 | }
406 |
407 | if (ignorePostMessage) {
408 | form.appendChild(internal.createFormHiddenInput("MERCHANT_RESPONSE_URL", redirectUrl));
409 | } else {
410 | var parser = internal.getUrlParser(window.location.href);
411 | var hppOriginParam = parser.protocol + '//' + parser.host;
412 |
413 | form.appendChild(internal.createFormHiddenInput("HPP_POST_RESPONSE", hppOriginParam));
414 | form.appendChild(internal.createFormHiddenInput("HPP_POST_DIMENSIONS", hppOriginParam));
415 | }
416 | return form;
417 | },
418 |
419 | /**
420 | * Creates a visual spinner element to be shown with the lightbox overlay while the
421 | * HPP's iframe loads
422 | *
423 | * @returns the created spinner element
424 | */
425 | createSpinner: function () {
426 | var spinner = document.createElement("img");
427 | spinner.setAttribute("src", "");
428 | spinner.setAttribute("id", "rxp-loader-" + randomId);
429 | spinner.style.left = "50%";
430 | spinner.style.position = "fixed";
431 | spinner.style.background = "#FFFFFF";
432 | spinner.style.borderRadius = "50%";
433 | spinner.style.width = "30px";
434 | spinner.style.zIndex = "200";
435 | spinner.style.marginLeft = "-15px";
436 | spinner.style.top = "120px";
437 | return spinner;
438 | },
439 |
440 | /**
441 | * Creates the HPP's form, spinner, iframe, and close button, appends them
442 | * to the DOM, and submits the form to load the HPP
443 | *
444 | * @param {HTMLDivElement} overlayElement
445 | * @param {object} token The HPP request data
446 | * @returns an object with the created spinner, iframe, and close button
447 | */
448 | createIFrame: function (overlayElement, token) {
449 | //Create the spinner
450 | var spinner = internal.createSpinner();
451 | document.body.appendChild(spinner);
452 |
453 | //Create the iframe
454 | var iFrame = document.createElement("iframe");
455 | iFrame.setAttribute("name", "rxp-frame-" + randomId);
456 | iFrame.setAttribute("id", "rxp-frame-" + randomId);
457 | iFrame.setAttribute("height", "562px");
458 | iFrame.setAttribute("frameBorder", "0");
459 | iFrame.setAttribute("width", "360px");
460 | iFrame.setAttribute("seamless", "seamless");
461 | iFrame.setAttribute("allow", "payment " + internal.getBaseUrl(hppUrl));
462 |
463 | iFrame.style.zIndex = "10001";
464 | iFrame.style.position = "absolute";
465 | iFrame.style.transition = "transform 0.5s ease-in-out";
466 | iFrame.style.transform = "scale(0.7)";
467 | iFrame.style.opacity = "0";
468 |
469 | overlayElement.appendChild(iFrame);
470 |
471 | if (isMobileIFrame) {
472 | iFrame.style.top = "0px";
473 | iFrame.style.bottom = "0px";
474 | iFrame.style.left = "0px";
475 | iFrame.style.marginLeft = "0px;";
476 | iFrame.style.width = "100%";
477 | iFrame.style.height = "100%";
478 | iFrame.style.minHeight = "100%";
479 | iFrame.style.WebkitTransform = "translate3d(0,0,0)";
480 | iFrame.style.transform = "translate3d(0, 0, 0)";
481 |
482 | var metaTag = document.createElement('meta');
483 | metaTag.name = "viewport";
484 | metaTag.content = "width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=0";
485 | document.getElementsByTagName('head')[0].appendChild(metaTag);
486 | } else {
487 | iFrame.style.top = "40px";
488 | iFrame.style.left = "50%";
489 | iFrame.style.marginLeft = "-180px";
490 | }
491 |
492 | logEvent(eventMessages.iFrame.create, { iFrame: iFrame });
493 |
494 | var closeButton;
495 |
496 | iFrame.onload = function () {
497 | iFrame.style.opacity = "1";
498 | iFrame.style.transform = "scale(1)";
499 | iFrame.style.backgroundColor = "#ffffff";
500 |
501 | if (spinner && spinner.parentNode) {
502 | spinner.parentNode.removeChild(spinner);
503 | }
504 |
505 | closeButton = internal.createCloseButton();
506 |
507 | if (overlayElement && closeButton) {
508 | overlayElement.appendChild(closeButton);
509 | closeButton.addEventListener("click", function () {
510 | internal.closeModal(closeButton, iFrame, spinner, overlayElement);
511 | }, true);
512 | }
513 | };
514 |
515 | var form = internal.createForm(document, token);
516 | logEvent(eventMessages.form.create, { form: form });
517 | if (iFrame.contentWindow.document.body) {
518 | iFrame.contentWindow.document.body.appendChild(form);
519 | logEvent(eventMessages.form.append);
520 | } else {
521 | iFrame.contentWindow.document.appendChild(form);
522 | logEvent(eventMessages.form.append);
523 | }
524 |
525 | form.submit();
526 | logEvent(eventMessages.form.submit);
527 |
528 | return {
529 | spinner: spinner,
530 | iFrame: iFrame,
531 | closeButton: closeButton
532 | };
533 | },
534 |
535 | /**
536 | * Opens the HPP in a new window
537 | *
538 | * Used in some mobile scenarios or when the browser viewport is
539 | * smaller than the HPP's inner width.
540 | *
541 | * Will automatically post the request data to the defined HPP
542 | * URL to load the HPP.
543 | *
544 | * @param {object} token The HPP request data
545 | * @returns the created window
546 | */
547 | openWindow: function (token) {
548 | //open new window
549 | var tabWindow = window.open();
550 |
551 | // browsers can prevent a new window from being created
552 | // e.g. mobile Safari
553 | if (!tabWindow) {
554 | return null;
555 | }
556 |
557 | var doc = tabWindow.document;
558 |
559 | //add meta tag to new window (needed for iOS 8 bug)
560 | var meta = doc.createElement("meta");
561 | var name = doc.createAttribute("name");
562 | name.value = "viewport";
563 | meta.setAttributeNode(name);
564 | var content = doc.createAttribute("content");
565 | content.value = "width=device-width";
566 | meta.setAttributeNode(content);
567 | doc.head.appendChild(meta);
568 |
569 | //create form, append to new window and submit
570 | var form = internal.createForm(doc, token);
571 | doc.body.appendChild(form);
572 | form.submit();
573 |
574 | return tabWindow;
575 | },
576 |
577 | /**
578 | * Creates a rudimentary URL parser using an anchor element
579 | *
580 | * @param {string} url
581 | * @returns the created anchor element
582 | */
583 | getUrlParser: function (url) {
584 | var parser = document.createElement('a');
585 | parser.href = url;
586 | return parser;
587 | },
588 |
589 | /**
590 | * Gets the hostname/origin from a URL. Used for origin checks
591 | *
592 | * @param {string} url
593 | * @returns the hostname/origin of the URL
594 | */
595 | getHostnameFromUrl: function (url) {
596 | return internal.getUrlParser(url).hostname;
597 | },
598 |
599 | /**
600 | * Gets the base URL from a URL. Used to set the 'allow payment' attribute.
601 | *
602 | * @param {string} url
603 | * @returns the base URL of the provided URL
604 | */
605 | getBaseUrl: function (url) {
606 | var urlParser = internal.getUrlParser(url);
607 | return urlParser.protocol + '//' + urlParser.hostname;
608 | },
609 |
610 | /**
611 | * Checks if the origin is HPP.
612 | *
613 | * @param {string} origin
614 | * @returns {boolean}
615 | */
616 | isHppOrigin: function(origin) {
617 | var result = false;
618 | allowedHppUrls.forEach(function (url) {
619 | if (internal.getHostnameFromUrl(url) === origin) {
620 | result = true;
621 | }
622 | });
623 |
624 | return result;
625 | },
626 |
627 | /**
628 | * Compares the origins from both arguments to validate we have received a postMessage
629 | * from the expected source
630 | *
631 | * @param {string} origin The origin attached to the recieved message
632 | * @param {string} hppUrl Our expected source origin
633 | * @returns true if the origins match
634 | */
635 | isMessageFromHpp: function (origin, hppUrl) {
636 | var originHostName = internal.getHostnameFromUrl(origin);
637 | return originHostName === internal.getHostnameFromUrl(hppUrl) || internal.isHppOrigin(originHostName);
638 | },
639 |
640 | /**
641 | * Handles messages from the HPP
642 | *
643 | * Messages from the HPP are one of:
644 | *
645 | * - iframe resize event
646 | * - transaction response
647 | * - error information
648 | *
649 | * @param {MessageEvent} e
650 | */
651 | receiveMessage: function (e) {
652 | //Check the origin of the response comes from HPP
653 | if (!internal.isMessageFromHpp(e.event.origin, hppUrl)) {
654 | return;
655 | }
656 |
657 | if (!e.event.data) {
658 | return;
659 | }
660 |
661 | var evtdata = internal.decodeAnswer(e.event.data);
662 |
663 | // we received an invalid message from the HPP iframe (e.g. from a browser plugin)
664 | // return early to prevent invalid processing
665 | if (evtdata === null) {
666 | return;
667 | }
668 |
669 | // check for iframe resize values
670 | if (evtdata.iframe) {
671 | if (!isMobileNewTab()) {
672 | var iframeWidth = evtdata.iframe.width;
673 | var iframeHeight = evtdata.iframe.height;
674 |
675 | var iFrame;
676 | var resized = false;
677 |
678 | if (e.embedded) {
679 | iFrame = e.instance.getIframe();
680 | } else {
681 | iFrame = document.getElementById("rxp-frame-" + randomId);
682 | }
683 | if (e.instance.events && e.instance.events.onResize) {
684 | e.instance.events.onResize(evtdata.iframe);
685 | }
686 |
687 | if (iframeWidth === "390px" && iframeHeight === "440px") {
688 | iFrame.setAttribute("width", iframeWidth);
689 | iFrame.setAttribute("height", iframeHeight);
690 | resized = true;
691 | }
692 |
693 | iFrame.style.backgroundColor="#ffffff";
694 |
695 | if (isMobileIFrame) {
696 | iFrame.style.marginLeft = "0px";
697 | iFrame.style.WebkitOverflowScrolling = "touch";
698 | iFrame.style.overflowX = "scroll";
699 | iFrame.style.overflowY = "scroll";
700 |
701 | if (!e.embedded) {
702 | var overlay = document.getElementById("rxp-overlay-" + randomId);
703 | overlay.style.overflowX = "scroll";
704 | overlay.style.overflowY = "scroll";
705 | }
706 | } else if (!e.embedded && resized) {
707 | iFrame.style.marginLeft = (parseInt(iframeWidth.replace("px", ""), 10) / 2 * -1) + "px";
708 | }
709 |
710 | if (!e.embedded && resized) {
711 | // wrap the below in a setTimeout to prevent a timing issue on a
712 | // cache-miss load
713 | setTimeout(function () {
714 | var closeButton = document.getElementById("rxp-frame-close-" + randomId);
715 | closeButton.style.marginLeft = ((parseInt(iframeWidth.replace("px", ""), 10) / 2) -7) + "px";
716 | }, 200);
717 | }
718 | }
719 | } else {
720 | var _close=function(){
721 | if (isMobileNewTab() && tabWindow) {
722 | //Close the new window
723 | tabWindow.close();
724 | } else {
725 | //Close the lightbox
726 | e.instance.close();
727 | }
728 | var overlay=document.getElementById("rxp-overlay-" + randomId);
729 | if(overlay) {
730 | overlay.remove();
731 | }
732 |
733 | };
734 | var response = e.event.data;
735 | //allow the script to intercept the answer, instead of redirecting to another page. (which is really a 90s thing)
736 | if (typeof e.url === 'function'){
737 | e.url(evtdata, _close);
738 | return;
739 | }
740 | _close();
741 | //Create a form and submit the hpp response to the merchant's response url
742 | var form = document.createElement("form");
743 | form.setAttribute("method", "POST");
744 | form.setAttribute("action", e.url);
745 | form.appendChild(internal.createFormHiddenInput("hppResponse", response));
746 | document.body.appendChild(form);
747 | form.submit();
748 | }
749 | }
750 | };
751 |
752 | /**
753 | * Public interface for the lightbox display mode
754 | */
755 | var RxpLightbox = (function () {
756 | var instance;
757 |
758 | function init() {
759 | var overlayElement;
760 | var spinner;
761 | var iFrame;
762 | var closeButton;
763 | var token;
764 | var isLandscape = internal.checkDevicesOrientation();
765 |
766 | if (isMobileIFrame) {
767 | if (window.addEventListener) {
768 | window.addEventListener("orientationchange", function () {
769 | isLandscape = internal.checkDevicesOrientation();
770 | }, false);
771 | }
772 | }
773 |
774 | return {
775 | lightbox: function () {
776 | if (isMobileNewTab()) {
777 | tabWindow = internal.openWindow(token);
778 | } else {
779 | overlayElement = internal.createOverlay();
780 | var temp = internal.createIFrame(overlayElement, token);
781 | spinner = temp.spinner;
782 | iFrame = temp.iFrame;
783 | closeButton = temp.closeButton;
784 | }
785 | },
786 | close: function () {
787 | internal.closeModal();
788 | },
789 | setToken: function (hppToken) {
790 | token = hppToken;
791 | }
792 | };
793 | }
794 |
795 | return {
796 | // Get the Singleton instance if one exists
797 | // or create one if it doesn't
798 | getInstance: function (hppToken) {
799 | if (!instance) {
800 | instance = init();
801 | }
802 |
803 | //Set the hpp token
804 | instance.setToken(hppToken);
805 |
806 | return instance;
807 | },
808 | init: function (idOfLightboxButton, merchantUrl, serverSdkJson) {
809 | logEvent(eventMessages.initialize('RxpLightbox'));
810 | //Get the lightbox instance (it's a singleton) and set the sdk json
811 | var lightboxInstance = RxpLightbox.getInstance(serverSdkJson);
812 |
813 | //if you want the form to load on function call, set to autoload
814 | if (idOfLightboxButton === 'autoload') {
815 | lightboxInstance.lightbox();
816 | }
817 | // Sets the event listener on the PAY button. The click will invoke the lightbox method
818 | else if (document.getElementById(idOfLightboxButton).addEventListener) {
819 | document.getElementById(idOfLightboxButton).addEventListener("click", lightboxInstance.lightbox, true);
820 | } else {
821 | document.getElementById(idOfLightboxButton).attachEvent('onclick', lightboxInstance.lightbox);
822 | }
823 | //avoid multiple message event listener binded to the window object.
824 | internal.removeOldEvtMsgListener();
825 | var evtMsgFct = function (event) {
826 | return internal.receiveMessage({ event: event, instance: lightboxInstance, url: merchantUrl, embedded: false });
827 | };
828 | internal.evtMsg.push({ fct: evtMsgFct, opt: false });
829 | internal.addEvtMsgListener(evtMsgFct);
830 | }
831 | };
832 | })();
833 |
834 | /**
835 | * Public interface for the embedded display mode
836 | */
837 | var RxpEmbedded = (function () {
838 | var instance;
839 |
840 | function init() {
841 | var overlayElement;
842 | var spinner;
843 | var iFrame;
844 | var closeButton;
845 | var token;
846 |
847 | return {
848 | embedded: function () {
849 | var form = internal.createForm(document, token);
850 | logEvent(eventMessages.form.create, { form: form });
851 | if (iFrame) {
852 | logEvent(eventMessages.iFrame.find, { iFrame: iFrame });
853 | if (iFrame.contentWindow.document.body) {
854 | iFrame.contentWindow.document.body.appendChild(form);
855 | logEvent(eventMessages.form.append);
856 | } else {
857 | iFrame.contentWindow.document.appendChild(form);
858 | logEvent(eventMessages.form.append);
859 | }
860 | form.submit();
861 | logEvent(eventMessages.form.submit);
862 | iFrame.style.display = "inherit";
863 | }
864 | },
865 | close: function () {
866 | iFrame.style.display = "none";
867 | },
868 | setToken: function (hppToken) {
869 | token = hppToken;
870 | },
871 | setIframe: function (iframeId) {
872 | iFrame = document.getElementById(iframeId);
873 | if (iFrame) {
874 | iFrame.setAttribute("allow", "payment " + internal.getBaseUrl(hppUrl));
875 | }
876 | },
877 | getIframe: function () {
878 | return iFrame;
879 | }
880 | };
881 | }
882 |
883 | return {
884 | // Get the Singleton instance if one exists
885 | // or create one if it doesn't
886 | getInstance: function (hppToken) {
887 | if (!instance) {
888 | instance = init();
889 | }
890 |
891 | //Set the hpp token
892 | instance.setToken(hppToken);
893 | return instance;
894 | },
895 | init: function (idOfEmbeddedButton, idOfTargetIframe, merchantUrl, serverSdkJson,events) {
896 | logEvent(eventMessages.initialize('RxpEmbedded'));
897 |
898 | //Get the embedded instance (it's a singleton) and set the sdk json
899 | var embeddedInstance = RxpEmbedded.getInstance(serverSdkJson);
900 | embeddedInstance.events=events;
901 |
902 | embeddedInstance.setIframe(idOfTargetIframe);
903 | //if you want the form to load on function call, set to autoload
904 | if (idOfEmbeddedButton === 'autoload') {
905 | embeddedInstance.embedded();
906 | }
907 | // Sets the event listener on the PAY button. The click will invoke the embedded method
908 | else if (document.getElementById(idOfEmbeddedButton).addEventListener) {
909 | document.getElementById(idOfEmbeddedButton).addEventListener("click", embeddedInstance.embedded, true);
910 | } else {
911 | document.getElementById(idOfEmbeddedButton).attachEvent('onclick', embeddedInstance.embedded);
912 | }
913 |
914 | //avoid multiple message event listener binded to the window object.
915 | internal.removeOldEvtMsgListener();
916 | var evtMsgFct = function (event) {
917 | return internal.receiveMessage({ event: event, instance: embeddedInstance, url: merchantUrl, embedded: true });
918 | };
919 | internal.evtMsg.push({ fct: evtMsgFct, opt: false });
920 | internal.addEvtMsgListener(evtMsgFct);
921 | }
922 | };
923 | })();
924 |
925 | /**
926 | * Public interface for the redirect display mode
927 | */
928 | var RxpRedirect = (function () {
929 | var instance;
930 |
931 | function init() {
932 | var overlayElement;
933 | var spinner;
934 | var iFrame;
935 | var closeButton;
936 | var token;
937 | var isLandscape = internal.checkDevicesOrientation();
938 |
939 | if (isMobileIFrame) {
940 | if (window.addEventListener) {
941 | window.addEventListener("orientationchange", function () {
942 | isLandscape = internal.checkDevicesOrientation();
943 | }, false);
944 | }
945 | }
946 |
947 | return {
948 | redirect: function () {
949 | var form = internal.createForm(document, token, true);
950 | logEvent(eventMessages.form.create, { form: form });
951 | document.body.append(form);
952 | form.submit();
953 | logEvent(eventMessages.form.submit);
954 | },
955 | setToken: function (hppToken) {
956 | token = hppToken;
957 | }
958 | };
959 | }
960 | return {
961 | // Get the singleton instance if one exists
962 | // or create one if it doesn't
963 | getInstance: function (hppToken) {
964 | if (!instance) {
965 | instance = init();
966 | }
967 |
968 | // Set the hpp token
969 | instance.setToken(hppToken);
970 |
971 | return instance;
972 | },
973 | init: function (idOfButton, merchantUrl, serverSdkJson) {
974 | logEvent(eventMessages.initialize('RxpRedirect'));
975 |
976 | // Get the redirect instance (it's a singleton) and set the sdk json
977 | var redirectInstance = RxpRedirect.getInstance(serverSdkJson);
978 | redirectUrl = merchantUrl;
979 |
980 | // Sets the event listener on the PAY button. The click will invoke the redirect method
981 | if (document.getElementById(idOfButton).addEventListener) {
982 | document.getElementById(idOfButton).addEventListener("click", redirectInstance.redirect, true);
983 | } else {
984 | document.getElementById(idOfButton).attachEvent('onclick', redirectInstance.redirect);
985 | }
986 |
987 | //avoid multiple message event listener binded to the window object.
988 | internal.removeOldEvtMsgListener();
989 | var evtMsgFct = function (event) {
990 | return internal.receiveMessage({ event: event, instance: redirectInstance, url: merchantUrl, embedded: false });
991 | };
992 | internal.evtMsg.push({ fct: evtMsgFct, opt: false });
993 | internal.addEvtMsgListener(evtMsgFct);
994 | }
995 | };
996 | }());
997 |
998 | /**
999 | * Public interface for the Realex HPP library
1000 | */
1001 | return {
1002 | init: RxpLightbox.init,
1003 | lightbox: {
1004 | init: RxpLightbox.init
1005 | },
1006 | embedded: {
1007 | init: RxpEmbedded.init
1008 | },
1009 | redirect: {
1010 | init: RxpRedirect.init
1011 | },
1012 | setHppUrl: setHppUrl,
1013 | setMobileXSLowerBound: setMobileXSLowerBound,
1014 | setConfigItem: setConfigItem,
1015 | constants: constants,
1016 | _internal: internal
1017 | };
1018 |
1019 | }());
1020 | var RealexRemote = (function() {
1021 |
1022 | 'use strict';
1023 |
1024 | /*
1025 | * Validate Card Number. Returns true if card number valid. Only allows
1026 | * non-empty numeric values between 12 and 19 characters. A Luhn check is
1027 | * also run against the card number.
1028 | */
1029 | var validateCardNumber = function(cardNumber) {
1030 | // test numeric and length between 12 and 19
1031 | if (!/^\d{12,19}$/.test(cardNumber)) {
1032 | return false;
1033 | }
1034 |
1035 | // luhn check
1036 | var sum = 0;
1037 | var digit = 0;
1038 | var addend = 0;
1039 | var timesTwo = false;
1040 |
1041 | for (var i = cardNumber.length - 1; i >= 0; i--) {
1042 | digit = parseInt(cardNumber.substring(i, i + 1), 10);
1043 | if (timesTwo) {
1044 | addend = digit * 2;
1045 | if (addend > 9) {
1046 | addend -= 9;
1047 | }
1048 | } else {
1049 | addend = digit;
1050 | }
1051 | sum += addend;
1052 | timesTwo = !timesTwo;
1053 | }
1054 |
1055 | var modulus = sum % 10;
1056 | if (modulus !== 0) {
1057 | return false;
1058 | }
1059 |
1060 | return true;
1061 | };
1062 |
1063 | /*
1064 | * Validate Card Holder Name. Returns true if card holder valid. Only allows
1065 | * non-empty ISO/IEC 8859-1 values 100 characters or less.
1066 | */
1067 | var validateCardHolderName = function(cardHolderName) {
1068 | // test for undefined
1069 | if (!cardHolderName) {
1070 | return false;
1071 | }
1072 |
1073 | // test white space only
1074 | if (!cardHolderName.trim()) {
1075 | return false;
1076 | }
1077 |
1078 | // test ISO/IEC 8859-1 characters between 1 and 100
1079 | if (!/^[\u0020-\u007E\u00A0-\u00FF]{1,100}$/.test(cardHolderName)) {
1080 | return false;
1081 | }
1082 |
1083 | return true;
1084 | };
1085 |
1086 | /*
1087 | * Validate CVN. Applies to non-Amex card types. Only allows 3 numeric
1088 | * characters.
1089 | */
1090 | var validateCvn = function(cvn) {
1091 | // test numeric length 3
1092 | if (!/^\d{3}$/.test(cvn)) {
1093 | return false;
1094 | }
1095 |
1096 | return true;
1097 | };
1098 |
1099 | /*
1100 | * Validate Amex CVN. Applies to Amex card types. Only allows 4 numeric
1101 | * characters.
1102 | */
1103 | var validateAmexCvn = function(cvn) {
1104 | // test numeric length 4
1105 | if (!/^\d{4}$/.test(cvn)) {
1106 | return false;
1107 | }
1108 |
1109 | return true;
1110 | };
1111 |
1112 | /*
1113 | * Validate Expiry Date Format. Only allows 4 numeric characters. Month must
1114 | * be between 1 and 12.
1115 | */
1116 | var validateExpiryDateFormat = function(expiryDate) {
1117 |
1118 | // test numeric of length 4
1119 | if (!/^\d{4}$/.test(expiryDate)) {
1120 | return false;
1121 | }
1122 |
1123 | var month = parseInt(expiryDate.substring(0, 2), 10);
1124 | var year = parseInt(expiryDate.substring(2, 4), 10);
1125 |
1126 | // test month range is 1-12
1127 | if (month < 1 || month > 12) {
1128 | return false;
1129 | }
1130 |
1131 | return true;
1132 | };
1133 |
1134 | /*
1135 | * Validate Expiry Date Not In Past. Also runs checks from
1136 | * validateExpiryDateFormat.
1137 | */
1138 | var validateExpiryDateNotInPast = function(expiryDate) {
1139 | // test valid format
1140 | if (!validateExpiryDateFormat(expiryDate)) {
1141 | return false;
1142 | }
1143 |
1144 | var month = parseInt(expiryDate.substring(0, 2), 10);
1145 | var year = parseInt(expiryDate.substring(2, 4), 10);
1146 |
1147 | // test date is not in the past
1148 | var now = new Date();
1149 | var currentMonth = now.getMonth() + 1;
1150 | var currentYear = now.getFullYear();
1151 | if (year < (currentYear % 100)) {
1152 | return false;
1153 | } else if (year === (currentYear % 100) && month < currentMonth) {
1154 | return false;
1155 | }
1156 |
1157 | return true;
1158 | };
1159 |
1160 | return {
1161 | validateCardNumber : validateCardNumber,
1162 | validateCardHolderName : validateCardHolderName,
1163 | validateCvn : validateCvn,
1164 | validateAmexCvn : validateAmexCvn,
1165 | validateExpiryDateFormat : validateExpiryDateFormat,
1166 | validateExpiryDateNotInPast : validateExpiryDateNotInPast
1167 | };
1168 | }());
1169 |
--------------------------------------------------------------------------------
/dist/rxp-js.min.js:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/globalpayments/rxp-js/f014bb09885a9d40d9b88c96eddf0df26e525088/dist/rxp-js.min.js
--------------------------------------------------------------------------------
/examples/hpp/helper.js:
--------------------------------------------------------------------------------
1 | function success(response) {
2 | console.log('Successful transaction. Message: ', response.MESSAGE);
3 | }
4 |
5 | $(document).ready(function() {
6 | $('#paymentMethod').on('change', function() {
7 | RealexHpp.setHppUrl(this.value);
8 | });
9 | });
10 |
--------------------------------------------------------------------------------
/examples/hpp/json/process-a-payment.json:
--------------------------------------------------------------------------------
1 | {
2 | "MERCHANT_ID": "heartlandgpsandbox",
3 | "ACCOUNT": "hpp",
4 | "AMOUNT": "1000",
5 | "CURRENCY": "EUR",
6 | "AUTO_SETTLE_FLAG": "1",
7 |
8 | "HPP_CUSTOMER_EMAIL": "test@example.com",
9 | "HPP_CUSTOMER_PHONENUMBER_MOBILE": "44|789456123",
10 | "HPP_BILLING_STREET1": "Flat 123",
11 | "HPP_BILLING_STREET2": "House 456",
12 | "HPP_BILLING_STREET3": "Unit 4",
13 | "HPP_BILLING_CITY": "Halifax",
14 | "HPP_BILLING_POSTALCODE": "W5 9HR",
15 | "HPP_BILLING_COUNTRY": "826",
16 | "HPP_SHIPPING_STREET1": "Apartment 852",
17 | "HPP_SHIPPING_STREET2": "Complex 741",
18 | "HPP_SHIPPING_STREET3": "House 963",
19 | "HPP_SHIPPING_CITY": "Chicago",
20 | "HPP_SHIPPING_STATE": "IL",
21 | "HPP_SHIPPING_POSTALCODE": "50001",
22 | "HPP_SHIPPING_COUNTRY": "840",
23 | "HPP_ADDRESS_MATCH_INDICATOR": "FALSE",
24 | "HPP_CHALLENGE_REQUEST_INDICATOR": "NO_PREFERENCE"
25 | }
--------------------------------------------------------------------------------
/examples/hpp/process-a-payment-embedded-autoload-callback.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | HPP embed Demo
5 |
6 |
12 |
13 |
14 |
15 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
--------------------------------------------------------------------------------
/examples/hpp/process-a-payment-embedded-autoload.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | HPP embed Demo
5 |
6 |
12 |
13 |
14 |
29 |
30 |
31 |
32 |
33 |
34 |
--------------------------------------------------------------------------------
/examples/hpp/process-a-payment-embedded.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | HPP embed Demo
5 |
6 |
12 |
13 |
14 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
--------------------------------------------------------------------------------
/examples/hpp/process-a-payment-lightbox-callback.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | HPP Lightbox Demo
5 |
6 |
7 |
8 |
9 |
37 |
38 |
39 |
40 |
41 |
45 |
46 |
47 |
48 |
49 |
50 |
--------------------------------------------------------------------------------
/examples/hpp/process-a-payment-lightbox.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | HPP Lightbox Demo
5 |
6 |
7 |
8 |
9 |
30 |
31 |
32 |
33 |
34 |
38 |
39 |
40 |
41 |
42 |
--------------------------------------------------------------------------------
/examples/hpp/proxy-request.php:
--------------------------------------------------------------------------------
1 | $value) {
66 | if (!$value) {
67 | continue;
68 | }
69 | $response[$key] = $value;
70 | }
71 |
72 | $response["ORDER_ID"] = substr(str_shuffle('abcdefghijklmnopqrstuvwxyz0123456789'), 0, 22);
73 | $response["TIMESTAMP"] = (new DateTime())->format("YmdHis");
74 | $response["SHA1HASH"] = generateHash($response, 'secret');
75 |
76 | $jsonResponse = json_encode($response);
77 |
78 | error_log('sending: ' . $jsonResponse);
79 | echo $jsonResponse;
80 |
--------------------------------------------------------------------------------
/examples/hpp/redirect-for-payment.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | HPP Redirect Demo
5 |
6 |
7 |
8 |
22 |
23 |
24 |
25 |
26 |
27 |
--------------------------------------------------------------------------------
/examples/hpp/response.php:
--------------------------------------------------------------------------------
1 | $v) {
18 | try {
19 | $hppResponse[$k] = base64_decode($v);
20 | } catch (Exception $e) {
21 | /* */
22 | }
23 | }
24 | }
25 | catch (Exception $e) {
26 | $hppResponse = $originalHppResponse;
27 | }
28 |
29 | ?>
30 |
31 |
32 | HPP Demo Response
33 |
34 |
35 |
36 | HPP Demo Response
37 |
40 |
43 |
44 |
45 |
--------------------------------------------------------------------------------
/lib/.jshintrc:
--------------------------------------------------------------------------------
1 | {
2 | "curly": true,
3 | "eqeqeq": true,
4 | "immed": true,
5 | "latedef": true,
6 | "newcap": true,
7 | "noarg": true,
8 | "sub": true,
9 | "undef": true,
10 | "boss": true,
11 | "eqnull": true,
12 | "predef": ["exports", "alert", "escape", "unescape"]
13 | }
14 |
--------------------------------------------------------------------------------
/lib/rxp-hpp.js:
--------------------------------------------------------------------------------
1 | /*jslint browser:true */
2 | Element.prototype.remove = function() {
3 | this.parentElement.removeChild(this);
4 | };
5 | NodeList.prototype.remove = HTMLCollection.prototype.remove = function() {
6 | for(var i = this.length - 1; i >= 0; i--) {
7 | if(this[i] && this[i].parentElement) {
8 | this[i].parentElement.removeChild(this[i]);
9 | }
10 | }
11 | };
12 | var RealexHpp = (function () {
13 |
14 | 'use strict';
15 |
16 | var hppUrl = "https://pay.realexpayments.com/pay";
17 |
18 | var allowedHppUrls = [
19 | 'https://pay.realexpayments.com/pay',
20 | 'https://pay.sandbox.realexpayments.com/pay'
21 | ];
22 |
23 | var randomId = randomId || Math.random().toString(16).substr(2,8);
24 |
25 | var setHppUrl = function(url) {
26 | hppUrl = url;
27 | };
28 |
29 | var mobileXSLowerBound = 360;
30 | var setMobileXSLowerBound = function (lowerBound) {
31 | mobileXSLowerBound = lowerBound;
32 | };
33 |
34 | var config = {
35 | enableLogging: false
36 | };
37 | var setConfigItem = function(configItem, value) {
38 | if (!config.hasOwnProperty(configItem)) {
39 | return;
40 | }
41 | config[configItem] = value;
42 | };
43 | var constants = {
44 | logEventName: 'rxp-log'
45 | };
46 | var eventMessages = {
47 | form: {
48 | append: 'Form appended to the iframe',
49 | create: 'Hidden form created',
50 | submit: 'Form submitted'
51 | },
52 | iFrame: {
53 | create: 'iFrame created',
54 | find: 'iFrame found'
55 | },
56 | initialize: function(mode) {
57 | return mode + ' initialized';
58 | }
59 | }
60 | var logEvent = function(message, data = {}) {
61 | if (!config.enableLogging) {
62 | return;
63 | }
64 |
65 | var event = new CustomEvent(constants.logEventName, { detail: { message: message, data: data } });
66 | window.dispatchEvent(event);
67 | };
68 |
69 | var isWindowsMobileOs = /Windows Phone|IEMobile/.test(navigator.userAgent);
70 | var isAndroidOrIOs = /Android|iPad|iPhone|iPod/.test(navigator.userAgent);
71 | var isMobileXS = function () {
72 | return (((window.innerWidth > 0) ? window.innerWidth : screen.width) <= mobileXSLowerBound ? true : false) ||
73 | (((window.innerHeight > 0) ? window.innerHeight : screen.Height) <= mobileXSLowerBound ? true : false);
74 | };
75 |
76 | // Display IFrame on WIndows Phone OS mobile devices
77 | var isMobileIFrame = isWindowsMobileOs;
78 |
79 | // For IOs/Android and small screen devices always open in new tab/window
80 | var isMobileNewTab = function () {
81 | return !isWindowsMobileOs && (isAndroidOrIOs || isMobileXS());
82 | };
83 |
84 | var tabWindow;
85 |
86 | var redirectUrl;
87 |
88 | /**
89 | * Shared functionality across lightbox, embedded, and redirect display modes.
90 | */
91 | var internal = {
92 | evtMsg: [],
93 | /**
94 | * Adds a new window message event listener and tracks it for later removal
95 | *
96 | * @param {Function} evtMsgFct
97 | */
98 | addEvtMsgListener: function(evtMsgFct) {
99 | this.evtMsg.push({ fct: evtMsgFct, opt: false });
100 | if (window.addEventListener) {
101 | window.addEventListener("message", evtMsgFct, false);
102 | } else {
103 | window.attachEvent('message', evtMsgFct);
104 | }
105 | },
106 | /**
107 | * Removes a previously set window message event listener
108 | */
109 | removeOldEvtMsgListener: function () {
110 | if (this.evtMsg.length > 0) {
111 | var evt = this.evtMsg.pop();
112 | if (window.addEventListener) {
113 | window.removeEventListener("message", evt.fct, evt.opt);
114 | } else {
115 | window.detachEvent('message', evt.fct);
116 | }
117 | }
118 | },
119 | /**
120 | * Shimmed base64 encode/decode support
121 | */
122 | base64:{
123 | encode:function(input) {
124 | var keyStr = "ABCDEFGHIJKLMNOP" +
125 | "QRSTUVWXYZabcdef" +
126 | "ghijklmnopqrstuv" +
127 | "wxyz0123456789+/" +
128 | "=";
129 | input = escape(input);
130 | var output = "";
131 | var chr1, chr2, chr3 = "";
132 | var enc1, enc2, enc3, enc4 = "";
133 | var i = 0;
134 |
135 | do {
136 | chr1 = input.charCodeAt(i++);
137 | chr2 = input.charCodeAt(i++);
138 | chr3 = input.charCodeAt(i++);
139 |
140 | enc1 = chr1 >> 2;
141 | enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);
142 | enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);
143 | enc4 = chr3 & 63;
144 |
145 | if (isNaN(chr2)) {
146 | enc3 = enc4 = 64;
147 | } else if (isNaN(chr3)) {
148 | enc4 = 64;
149 | }
150 |
151 | output = output +
152 | keyStr.charAt(enc1) +
153 | keyStr.charAt(enc2) +
154 | keyStr.charAt(enc3) +
155 | keyStr.charAt(enc4);
156 | chr1 = chr2 = chr3 = "";
157 | enc1 = enc2 = enc3 = enc4 = "";
158 | } while (i < input.length);
159 |
160 | return output;
161 | },
162 | decode:function(input) {
163 | if(typeof input === 'undefined') {
164 | return input;
165 | }
166 | var keyStr = "ABCDEFGHIJKLMNOP" +
167 | "QRSTUVWXYZabcdef" +
168 | "ghijklmnopqrstuv" +
169 | "wxyz0123456789+/" +
170 | "=";
171 | var output = "";
172 | var chr1, chr2, chr3 = "";
173 | var enc1, enc2, enc3, enc4 = "";
174 | var i = 0;
175 |
176 | // remove all characters that are not A-Z, a-z, 0-9, +, /, or =
177 | var base64test = /[^A-Za-z0-9\+\/\=]/g;
178 | if (base64test.exec(input)) {
179 | throw new Error("There were invalid base64 characters in the input text.\n" +
180 | "Valid base64 characters are A-Z, a-z, 0-9, '+', '/',and '='\n" +
181 | "Expect errors in decoding.");
182 | }
183 | input = input.replace(/[^A-Za-z0-9\+\/\=]/g, "");
184 |
185 | do {
186 | enc1 = keyStr.indexOf(input.charAt(i++));
187 | enc2 = keyStr.indexOf(input.charAt(i++));
188 | enc3 = keyStr.indexOf(input.charAt(i++));
189 | enc4 = keyStr.indexOf(input.charAt(i++));
190 |
191 | chr1 = (enc1 << 2) | (enc2 >> 4);
192 | chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);
193 | chr3 = ((enc3 & 3) << 6) | enc4;
194 |
195 | output = output + String.fromCharCode(chr1);
196 |
197 | if (enc3 !== 64) {
198 | output = output + String.fromCharCode(chr2);
199 | }
200 | if (enc4 !== 64) {
201 | output = output + String.fromCharCode(chr3);
202 | }
203 |
204 | chr1 = chr2 = chr3 = "";
205 | enc1 = enc2 = enc3 = enc4 = "";
206 |
207 | } while (i < input.length);
208 |
209 | return unescape(output);
210 | }
211 | },
212 | /**
213 | * Converts an HPP message to a developer-friendly version.
214 | *
215 | * The decode process has two steps:
216 | *
217 | * 1. Attempt to parse the string as JSON. If this fails, an error response
218 | * is provided as we expect that the HPP has errored out to the cardholder
219 | * 2. Attempt to base64 decode the data to cover both HPP versions 1 and 2.
220 | *
221 | * @param {any} answer
222 | * @returns null if answer is not a string, otherwise the data from the HPP
223 | */
224 | decodeAnswer:function(answer){ //internal.decodeAnswer
225 |
226 | var _r;
227 |
228 | if (typeof answer !== "string") {
229 | return null;
230 | }
231 |
232 | try {
233 | _r=JSON.parse(answer);
234 | } catch (e) {
235 | _r = { error: true, message: answer };
236 | }
237 |
238 | try {
239 | for(var r in _r){
240 | if(_r[r]) {
241 | _r[r]=internal.base64.decode(_r[r]);
242 | }
243 | }
244 | } catch (e) { /** */ }
245 | return _r;
246 | },
247 | /**
248 | * Creates a new input of type `hidden`. Does not append to DOM.
249 | *
250 | * @param {string} name Name for the new input
251 | * @param {string} value Value for the new input
252 | * @returns the created input
253 | */
254 | createFormHiddenInput: function (name, value) {
255 | var el = document.createElement("input");
256 | el.setAttribute("type", "hidden");
257 | el.setAttribute("name", name);
258 | el.setAttribute("value", value);
259 | return el;
260 | },
261 |
262 | /**
263 | * Determines a mobile device's orientation for width calculation
264 | *
265 | * @returns true if in landscape
266 | */
267 | checkDevicesOrientation: function () {
268 | if (window.orientation === 90 || window.orientation === -90) {
269 | return true;
270 | } else {
271 | return false;
272 | }
273 | },
274 |
275 | /**
276 | * Creates a semi-transparent overlay with full width/height to serve as
277 | * a background for the lightbox modal
278 | *
279 | * @returns the created overlay
280 | */
281 | createOverlay: function () {
282 | var overlay = document.createElement("div");
283 | overlay.setAttribute("id", "rxp-overlay-" + randomId);
284 | overlay.style.position = "fixed";
285 | overlay.style.width = "100%";
286 | overlay.style.height = "100%";
287 | overlay.style.top = "0";
288 | overlay.style.left = "0";
289 | overlay.style.transition = "all 0.3s ease-in-out";
290 | overlay.style.zIndex = "100";
291 |
292 | if (isMobileIFrame) {
293 | overlay.style.position = "absolute !important";
294 | overlay.style.WebkitOverflowScrolling = "touch";
295 | overlay.style.overflowX = "hidden";
296 | overlay.style.overflowY = "scroll";
297 | }
298 |
299 | document.body.appendChild(overlay);
300 |
301 | setTimeout(function () {
302 | overlay.style.background = "rgba(0, 0, 0, 0.7)";
303 | }, 1);
304 |
305 | return overlay;
306 | },
307 |
308 | /**
309 | * Closes a lightbox modal and all associated elements
310 | *
311 | * @param {HTMLImageElement} closeButton
312 | * @param {HTMLIFrameElement} iFrame
313 | * @param {HTMLImageElement} spinner
314 | * @param {HTMLDivElement} overlayElement
315 | */
316 | closeModal: function (closeButton, iFrame, spinner, overlayElement) {
317 | if (closeButton && closeButton.parentNode) {
318 | closeButton.parentNode.removeChild(closeButton);
319 | }
320 |
321 | if (iFrame && iFrame.parentNode) {
322 | iFrame.parentNode.removeChild(iFrame);
323 | }
324 |
325 | if (spinner && spinner.parentNode) {
326 | spinner.parentNode.removeChild(spinner);
327 | }
328 |
329 | if (!overlayElement) {
330 | return;
331 | }
332 |
333 | overlayElement.className = "";
334 | setTimeout(function () {
335 | if (overlayElement.parentNode) {
336 | overlayElement.parentNode.removeChild(overlayElement);
337 | }
338 | }, 300);
339 | },
340 |
341 | /**
342 | * Creates a close button for the lightbox modal
343 | *
344 | * @returns the created element
345 | */
346 | createCloseButton: function (overlayElement) {
347 | if (document.getElementById("rxp-frame-close-" + randomId) !== null) {
348 | return;
349 | }
350 |
351 | var closeButton = document.createElement("img");
352 | closeButton.setAttribute("id","rxp-frame-close-" + randomId);
353 | closeButton.setAttribute("src", "");
354 | closeButton.setAttribute("style","transition: all 0.5s ease-in-out; opacity: 0; float: left; position: absolute; left: 50%; margin-left: 173px; z-index: 99999999; top: 30px;");
355 |
356 | setTimeout(function () {
357 | closeButton.style.opacity = "1";
358 | },500);
359 |
360 | if (isMobileIFrame) {
361 | closeButton.style.position = "absolute";
362 | closeButton.style.float = "right";
363 | closeButton.style.top = "20px";
364 | closeButton.style.left = "initial";
365 | closeButton.style.marginLeft = "0px";
366 | closeButton.style.right = "20px";
367 | }
368 |
369 | return closeButton;
370 | },
371 |
372 | /**
373 | * Creates a form and appends the HPP request data as hidden input elements to
374 | * POST to the defined HPP URL.
375 | *
376 | * The created form is not appended to the DOM and is not submitted at this time.
377 | *
378 | * @param {Document} doc
379 | * @param {object} token HPP request data
380 | * @param {bool} ignorePostMessage If true, the HPP will redirect to the defined
381 | * defined redirect URL. If false, the HPP will send a postMessage
382 | * to the parent window to be handled by this library.
383 | * @returns the created form
384 | */
385 | createForm: function (doc, token, ignorePostMessage) {
386 | var form = document.createElement("form");
387 | form.setAttribute("method", "POST");
388 | form.setAttribute("action", hppUrl);
389 |
390 | var versionSet = false;
391 |
392 | for (var key in token) {
393 | if (key === "HPP_VERSION"){
394 | versionSet = true;
395 | }
396 | form.appendChild(internal.createFormHiddenInput(key, token[key]));
397 | }
398 |
399 | if (versionSet === false){
400 | form.appendChild(internal.createFormHiddenInput("HPP_VERSION", "2"));
401 | }
402 |
403 | if (ignorePostMessage) {
404 | form.appendChild(internal.createFormHiddenInput("MERCHANT_RESPONSE_URL", redirectUrl));
405 | } else {
406 | var parser = internal.getUrlParser(window.location.href);
407 | var hppOriginParam = parser.protocol + '//' + parser.host;
408 |
409 | form.appendChild(internal.createFormHiddenInput("HPP_POST_RESPONSE", hppOriginParam));
410 | form.appendChild(internal.createFormHiddenInput("HPP_POST_DIMENSIONS", hppOriginParam));
411 | }
412 | return form;
413 | },
414 |
415 | /**
416 | * Creates a visual spinner element to be shown with the lightbox overlay while the
417 | * HPP's iframe loads
418 | *
419 | * @returns the created spinner element
420 | */
421 | createSpinner: function () {
422 | var spinner = document.createElement("img");
423 | spinner.setAttribute("src", "");
424 | spinner.setAttribute("id", "rxp-loader-" + randomId);
425 | spinner.style.left = "50%";
426 | spinner.style.position = "fixed";
427 | spinner.style.background = "#FFFFFF";
428 | spinner.style.borderRadius = "50%";
429 | spinner.style.width = "30px";
430 | spinner.style.zIndex = "200";
431 | spinner.style.marginLeft = "-15px";
432 | spinner.style.top = "120px";
433 | return spinner;
434 | },
435 |
436 | /**
437 | * Creates the HPP's form, spinner, iframe, and close button, appends them
438 | * to the DOM, and submits the form to load the HPP
439 | *
440 | * @param {HTMLDivElement} overlayElement
441 | * @param {object} token The HPP request data
442 | * @returns an object with the created spinner, iframe, and close button
443 | */
444 | createIFrame: function (overlayElement, token) {
445 | //Create the spinner
446 | var spinner = internal.createSpinner();
447 | document.body.appendChild(spinner);
448 |
449 | //Create the iframe
450 | var iFrame = document.createElement("iframe");
451 | iFrame.setAttribute("name", "rxp-frame-" + randomId);
452 | iFrame.setAttribute("id", "rxp-frame-" + randomId);
453 | iFrame.setAttribute("height", "562px");
454 | iFrame.setAttribute("frameBorder", "0");
455 | iFrame.setAttribute("width", "360px");
456 | iFrame.setAttribute("seamless", "seamless");
457 | iFrame.setAttribute("allow", "payment " + internal.getBaseUrl(hppUrl));
458 |
459 | iFrame.style.zIndex = "10001";
460 | iFrame.style.position = "absolute";
461 | iFrame.style.transition = "transform 0.5s ease-in-out";
462 | iFrame.style.transform = "scale(0.7)";
463 | iFrame.style.opacity = "0";
464 |
465 | overlayElement.appendChild(iFrame);
466 |
467 | if (isMobileIFrame) {
468 | iFrame.style.top = "0px";
469 | iFrame.style.bottom = "0px";
470 | iFrame.style.left = "0px";
471 | iFrame.style.marginLeft = "0px;";
472 | iFrame.style.width = "100%";
473 | iFrame.style.height = "100%";
474 | iFrame.style.minHeight = "100%";
475 | iFrame.style.WebkitTransform = "translate3d(0,0,0)";
476 | iFrame.style.transform = "translate3d(0, 0, 0)";
477 |
478 | var metaTag = document.createElement('meta');
479 | metaTag.name = "viewport";
480 | metaTag.content = "width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=0";
481 | document.getElementsByTagName('head')[0].appendChild(metaTag);
482 | } else {
483 | iFrame.style.top = "40px";
484 | iFrame.style.left = "50%";
485 | iFrame.style.marginLeft = "-180px";
486 | }
487 |
488 | logEvent(eventMessages.iFrame.create, { iFrame: iFrame });
489 |
490 | var closeButton;
491 |
492 | iFrame.onload = function () {
493 | iFrame.style.opacity = "1";
494 | iFrame.style.transform = "scale(1)";
495 | iFrame.style.backgroundColor = "#ffffff";
496 |
497 | if (spinner && spinner.parentNode) {
498 | spinner.parentNode.removeChild(spinner);
499 | }
500 |
501 | closeButton = internal.createCloseButton();
502 |
503 | if (overlayElement && closeButton) {
504 | overlayElement.appendChild(closeButton);
505 | closeButton.addEventListener("click", function () {
506 | internal.closeModal(closeButton, iFrame, spinner, overlayElement);
507 | }, true);
508 | }
509 | };
510 |
511 | var form = internal.createForm(document, token);
512 | logEvent(eventMessages.form.create, { form: form });
513 | if (iFrame.contentWindow.document.body) {
514 | iFrame.contentWindow.document.body.appendChild(form);
515 | logEvent(eventMessages.form.append);
516 | } else {
517 | iFrame.contentWindow.document.appendChild(form);
518 | logEvent(eventMessages.form.append);
519 | }
520 |
521 | form.submit();
522 | logEvent(eventMessages.form.submit);
523 |
524 | return {
525 | spinner: spinner,
526 | iFrame: iFrame,
527 | closeButton: closeButton
528 | };
529 | },
530 |
531 | /**
532 | * Opens the HPP in a new window
533 | *
534 | * Used in some mobile scenarios or when the browser viewport is
535 | * smaller than the HPP's inner width.
536 | *
537 | * Will automatically post the request data to the defined HPP
538 | * URL to load the HPP.
539 | *
540 | * @param {object} token The HPP request data
541 | * @returns the created window
542 | */
543 | openWindow: function (token) {
544 | //open new window
545 | var tabWindow = window.open();
546 |
547 | // browsers can prevent a new window from being created
548 | // e.g. mobile Safari
549 | if (!tabWindow) {
550 | return null;
551 | }
552 |
553 | var doc = tabWindow.document;
554 |
555 | //add meta tag to new window (needed for iOS 8 bug)
556 | var meta = doc.createElement("meta");
557 | var name = doc.createAttribute("name");
558 | name.value = "viewport";
559 | meta.setAttributeNode(name);
560 | var content = doc.createAttribute("content");
561 | content.value = "width=device-width";
562 | meta.setAttributeNode(content);
563 | doc.head.appendChild(meta);
564 |
565 | //create form, append to new window and submit
566 | var form = internal.createForm(doc, token);
567 | doc.body.appendChild(form);
568 | form.submit();
569 |
570 | return tabWindow;
571 | },
572 |
573 | /**
574 | * Creates a rudimentary URL parser using an anchor element
575 | *
576 | * @param {string} url
577 | * @returns the created anchor element
578 | */
579 | getUrlParser: function (url) {
580 | var parser = document.createElement('a');
581 | parser.href = url;
582 | return parser;
583 | },
584 |
585 | /**
586 | * Gets the hostname/origin from a URL. Used for origin checks
587 | *
588 | * @param {string} url
589 | * @returns the hostname/origin of the URL
590 | */
591 | getHostnameFromUrl: function (url) {
592 | return internal.getUrlParser(url).hostname;
593 | },
594 |
595 | /**
596 | * Gets the base URL from a URL. Used to set the 'allow payment' attribute.
597 | *
598 | * @param {string} url
599 | * @returns the base URL of the provided URL
600 | */
601 | getBaseUrl: function (url) {
602 | var urlParser = internal.getUrlParser(url);
603 | return urlParser.protocol + '//' + urlParser.hostname;
604 | },
605 |
606 | /**
607 | * Checks if the origin is HPP.
608 | *
609 | * @param {string} origin
610 | * @returns {boolean}
611 | */
612 | isHppOrigin: function(origin) {
613 | var result = false;
614 | allowedHppUrls.forEach(function (url) {
615 | if (internal.getHostnameFromUrl(url) === origin) {
616 | result = true;
617 | }
618 | });
619 |
620 | return result;
621 | },
622 |
623 | /**
624 | * Compares the origins from both arguments to validate we have received a postMessage
625 | * from the expected source
626 | *
627 | * @param {string} origin The origin attached to the recieved message
628 | * @param {string} hppUrl Our expected source origin
629 | * @returns true if the origins match
630 | */
631 | isMessageFromHpp: function (origin, hppUrl) {
632 | var originHostName = internal.getHostnameFromUrl(origin);
633 | return originHostName === internal.getHostnameFromUrl(hppUrl) || internal.isHppOrigin(originHostName);
634 | },
635 |
636 | /**
637 | * Handles messages from the HPP
638 | *
639 | * Messages from the HPP are one of:
640 | *
641 | * - iframe resize event
642 | * - transaction response
643 | * - error information
644 | *
645 | * @param {MessageEvent} e
646 | */
647 | receiveMessage: function (e) {
648 | //Check the origin of the response comes from HPP
649 | if (!internal.isMessageFromHpp(e.event.origin, hppUrl)) {
650 | return;
651 | }
652 |
653 | if (!e.event.data) {
654 | return;
655 | }
656 |
657 | var evtdata = internal.decodeAnswer(e.event.data);
658 |
659 | // we received an invalid message from the HPP iframe (e.g. from a browser plugin)
660 | // return early to prevent invalid processing
661 | if (evtdata === null) {
662 | return;
663 | }
664 |
665 | // check for iframe resize values
666 | if (evtdata.iframe) {
667 | if (!isMobileNewTab()) {
668 | var iframeWidth = evtdata.iframe.width;
669 | var iframeHeight = evtdata.iframe.height;
670 |
671 | var iFrame;
672 | var resized = false;
673 |
674 | if (e.embedded) {
675 | iFrame = e.instance.getIframe();
676 | } else {
677 | iFrame = document.getElementById("rxp-frame-" + randomId);
678 | }
679 | if (e.instance.events && e.instance.events.onResize) {
680 | e.instance.events.onResize(evtdata.iframe);
681 | }
682 |
683 | if (iframeWidth === "390px" && iframeHeight === "440px") {
684 | iFrame.setAttribute("width", iframeWidth);
685 | iFrame.setAttribute("height", iframeHeight);
686 | resized = true;
687 | }
688 |
689 | iFrame.style.backgroundColor="#ffffff";
690 |
691 | if (isMobileIFrame) {
692 | iFrame.style.marginLeft = "0px";
693 | iFrame.style.WebkitOverflowScrolling = "touch";
694 | iFrame.style.overflowX = "scroll";
695 | iFrame.style.overflowY = "scroll";
696 |
697 | if (!e.embedded) {
698 | var overlay = document.getElementById("rxp-overlay-" + randomId);
699 | overlay.style.overflowX = "scroll";
700 | overlay.style.overflowY = "scroll";
701 | }
702 | } else if (!e.embedded && resized) {
703 | iFrame.style.marginLeft = (parseInt(iframeWidth.replace("px", ""), 10) / 2 * -1) + "px";
704 | }
705 |
706 | if (!e.embedded && resized) {
707 | // wrap the below in a setTimeout to prevent a timing issue on a
708 | // cache-miss load
709 | setTimeout(function () {
710 | var closeButton = document.getElementById("rxp-frame-close-" + randomId);
711 | closeButton.style.marginLeft = ((parseInt(iframeWidth.replace("px", ""), 10) / 2) -7) + "px";
712 | }, 200);
713 | }
714 | }
715 | } else {
716 | var _close=function(){
717 | if (isMobileNewTab() && tabWindow) {
718 | //Close the new window
719 | tabWindow.close();
720 | } else {
721 | //Close the lightbox
722 | e.instance.close();
723 | }
724 | var overlay=document.getElementById("rxp-overlay-" + randomId);
725 | if(overlay) {
726 | overlay.remove();
727 | }
728 |
729 | };
730 | var response = e.event.data;
731 | //allow the script to intercept the answer, instead of redirecting to another page. (which is really a 90s thing)
732 | if (typeof e.url === 'function'){
733 | e.url(evtdata, _close);
734 | return;
735 | }
736 | _close();
737 | //Create a form and submit the hpp response to the merchant's response url
738 | var form = document.createElement("form");
739 | form.setAttribute("method", "POST");
740 | form.setAttribute("action", e.url);
741 | form.appendChild(internal.createFormHiddenInput("hppResponse", response));
742 | document.body.appendChild(form);
743 | form.submit();
744 | }
745 | }
746 | };
747 |
748 | /**
749 | * Public interface for the lightbox display mode
750 | */
751 | var RxpLightbox = (function () {
752 | var instance;
753 |
754 | function init() {
755 | var overlayElement;
756 | var spinner;
757 | var iFrame;
758 | var closeButton;
759 | var token;
760 | var isLandscape = internal.checkDevicesOrientation();
761 |
762 | if (isMobileIFrame) {
763 | if (window.addEventListener) {
764 | window.addEventListener("orientationchange", function () {
765 | isLandscape = internal.checkDevicesOrientation();
766 | }, false);
767 | }
768 | }
769 |
770 | return {
771 | lightbox: function () {
772 | if (isMobileNewTab()) {
773 | tabWindow = internal.openWindow(token);
774 | } else {
775 | overlayElement = internal.createOverlay();
776 | var temp = internal.createIFrame(overlayElement, token);
777 | spinner = temp.spinner;
778 | iFrame = temp.iFrame;
779 | closeButton = temp.closeButton;
780 | }
781 | },
782 | close: function () {
783 | internal.closeModal();
784 | },
785 | setToken: function (hppToken) {
786 | token = hppToken;
787 | }
788 | };
789 | }
790 |
791 | return {
792 | // Get the Singleton instance if one exists
793 | // or create one if it doesn't
794 | getInstance: function (hppToken) {
795 | if (!instance) {
796 | instance = init();
797 | }
798 |
799 | //Set the hpp token
800 | instance.setToken(hppToken);
801 |
802 | return instance;
803 | },
804 | init: function (idOfLightboxButton, merchantUrl, serverSdkJson) {
805 | logEvent(eventMessages.initialize('RxpLightbox'));
806 | //Get the lightbox instance (it's a singleton) and set the sdk json
807 | var lightboxInstance = RxpLightbox.getInstance(serverSdkJson);
808 |
809 | //if you want the form to load on function call, set to autoload
810 | if (idOfLightboxButton === 'autoload') {
811 | lightboxInstance.lightbox();
812 | }
813 | // Sets the event listener on the PAY button. The click will invoke the lightbox method
814 | else if (document.getElementById(idOfLightboxButton).addEventListener) {
815 | document.getElementById(idOfLightboxButton).addEventListener("click", lightboxInstance.lightbox, true);
816 | } else {
817 | document.getElementById(idOfLightboxButton).attachEvent('onclick', lightboxInstance.lightbox);
818 | }
819 | //avoid multiple message event listener binded to the window object.
820 | internal.removeOldEvtMsgListener();
821 | var evtMsgFct = function (event) {
822 | return internal.receiveMessage({ event: event, instance: lightboxInstance, url: merchantUrl, embedded: false });
823 | };
824 | internal.evtMsg.push({ fct: evtMsgFct, opt: false });
825 | internal.addEvtMsgListener(evtMsgFct);
826 | }
827 | };
828 | })();
829 |
830 | /**
831 | * Public interface for the embedded display mode
832 | */
833 | var RxpEmbedded = (function () {
834 | var instance;
835 |
836 | function init() {
837 | var overlayElement;
838 | var spinner;
839 | var iFrame;
840 | var closeButton;
841 | var token;
842 |
843 | return {
844 | embedded: function () {
845 | var form = internal.createForm(document, token);
846 | logEvent(eventMessages.form.create, { form: form });
847 | if (iFrame) {
848 | logEvent(eventMessages.iFrame.find, { iFrame: iFrame });
849 | if (iFrame.contentWindow.document.body) {
850 | iFrame.contentWindow.document.body.appendChild(form);
851 | logEvent(eventMessages.form.append);
852 | } else {
853 | iFrame.contentWindow.document.appendChild(form);
854 | logEvent(eventMessages.form.append);
855 | }
856 | form.submit();
857 | logEvent(eventMessages.form.submit);
858 | iFrame.style.display = "inherit";
859 | }
860 | },
861 | close: function () {
862 | iFrame.style.display = "none";
863 | },
864 | setToken: function (hppToken) {
865 | token = hppToken;
866 | },
867 | setIframe: function (iframeId) {
868 | iFrame = document.getElementById(iframeId);
869 | if (iFrame) {
870 | iFrame.setAttribute("allow", "payment " + internal.getBaseUrl(hppUrl));
871 | }
872 | },
873 | getIframe: function () {
874 | return iFrame;
875 | }
876 | };
877 | }
878 |
879 | return {
880 | // Get the Singleton instance if one exists
881 | // or create one if it doesn't
882 | getInstance: function (hppToken) {
883 | if (!instance) {
884 | instance = init();
885 | }
886 |
887 | //Set the hpp token
888 | instance.setToken(hppToken);
889 | return instance;
890 | },
891 | init: function (idOfEmbeddedButton, idOfTargetIframe, merchantUrl, serverSdkJson,events) {
892 | logEvent(eventMessages.initialize('RxpEmbedded'));
893 |
894 | //Get the embedded instance (it's a singleton) and set the sdk json
895 | var embeddedInstance = RxpEmbedded.getInstance(serverSdkJson);
896 | embeddedInstance.events=events;
897 |
898 | embeddedInstance.setIframe(idOfTargetIframe);
899 | //if you want the form to load on function call, set to autoload
900 | if (idOfEmbeddedButton === 'autoload') {
901 | embeddedInstance.embedded();
902 | }
903 | // Sets the event listener on the PAY button. The click will invoke the embedded method
904 | else if (document.getElementById(idOfEmbeddedButton).addEventListener) {
905 | document.getElementById(idOfEmbeddedButton).addEventListener("click", embeddedInstance.embedded, true);
906 | } else {
907 | document.getElementById(idOfEmbeddedButton).attachEvent('onclick', embeddedInstance.embedded);
908 | }
909 |
910 | //avoid multiple message event listener binded to the window object.
911 | internal.removeOldEvtMsgListener();
912 | var evtMsgFct = function (event) {
913 | return internal.receiveMessage({ event: event, instance: embeddedInstance, url: merchantUrl, embedded: true });
914 | };
915 | internal.evtMsg.push({ fct: evtMsgFct, opt: false });
916 | internal.addEvtMsgListener(evtMsgFct);
917 | }
918 | };
919 | })();
920 |
921 | /**
922 | * Public interface for the redirect display mode
923 | */
924 | var RxpRedirect = (function () {
925 | var instance;
926 |
927 | function init() {
928 | var overlayElement;
929 | var spinner;
930 | var iFrame;
931 | var closeButton;
932 | var token;
933 | var isLandscape = internal.checkDevicesOrientation();
934 |
935 | if (isMobileIFrame) {
936 | if (window.addEventListener) {
937 | window.addEventListener("orientationchange", function () {
938 | isLandscape = internal.checkDevicesOrientation();
939 | }, false);
940 | }
941 | }
942 |
943 | return {
944 | redirect: function () {
945 | var form = internal.createForm(document, token, true);
946 | logEvent(eventMessages.form.create, { form: form });
947 | document.body.append(form);
948 | form.submit();
949 | logEvent(eventMessages.form.submit);
950 | },
951 | setToken: function (hppToken) {
952 | token = hppToken;
953 | }
954 | };
955 | }
956 | return {
957 | // Get the singleton instance if one exists
958 | // or create one if it doesn't
959 | getInstance: function (hppToken) {
960 | if (!instance) {
961 | instance = init();
962 | }
963 |
964 | // Set the hpp token
965 | instance.setToken(hppToken);
966 |
967 | return instance;
968 | },
969 | init: function (idOfButton, merchantUrl, serverSdkJson) {
970 | logEvent(eventMessages.initialize('RxpRedirect'));
971 |
972 | // Get the redirect instance (it's a singleton) and set the sdk json
973 | var redirectInstance = RxpRedirect.getInstance(serverSdkJson);
974 | redirectUrl = merchantUrl;
975 |
976 | // Sets the event listener on the PAY button. The click will invoke the redirect method
977 | if (document.getElementById(idOfButton).addEventListener) {
978 | document.getElementById(idOfButton).addEventListener("click", redirectInstance.redirect, true);
979 | } else {
980 | document.getElementById(idOfButton).attachEvent('onclick', redirectInstance.redirect);
981 | }
982 |
983 | //avoid multiple message event listener binded to the window object.
984 | internal.removeOldEvtMsgListener();
985 | var evtMsgFct = function (event) {
986 | return internal.receiveMessage({ event: event, instance: redirectInstance, url: merchantUrl, embedded: false });
987 | };
988 | internal.evtMsg.push({ fct: evtMsgFct, opt: false });
989 | internal.addEvtMsgListener(evtMsgFct);
990 | }
991 | };
992 | }());
993 |
994 | /**
995 | * Public interface for the Realex HPP library
996 | */
997 | return {
998 | init: RxpLightbox.init,
999 | lightbox: {
1000 | init: RxpLightbox.init
1001 | },
1002 | embedded: {
1003 | init: RxpEmbedded.init
1004 | },
1005 | redirect: {
1006 | init: RxpRedirect.init
1007 | },
1008 | setHppUrl: setHppUrl,
1009 | setMobileXSLowerBound: setMobileXSLowerBound,
1010 | setConfigItem: setConfigItem,
1011 | constants: constants,
1012 | _internal: internal
1013 | };
1014 |
1015 | }());
--------------------------------------------------------------------------------
/lib/rxp-remote.js:
--------------------------------------------------------------------------------
1 | /*
2 | * rxp-remote.js
3 | * https://github.com/realexpayments/rxp-js
4 | *
5 | * Licensed under the MIT license.
6 | */
7 | var RealexRemote = (function() {
8 |
9 | 'use strict';
10 |
11 | /*
12 | * Validate Card Number. Returns true if card number valid. Only allows
13 | * non-empty numeric values between 12 and 19 characters. A Luhn check is
14 | * also run against the card number.
15 | */
16 | var validateCardNumber = function(cardNumber) {
17 | // test numeric and length between 12 and 19
18 | if (!/^\d{12,19}$/.test(cardNumber)) {
19 | return false;
20 | }
21 |
22 | // luhn check
23 | var sum = 0;
24 | var digit = 0;
25 | var addend = 0;
26 | var timesTwo = false;
27 |
28 | for (var i = cardNumber.length - 1; i >= 0; i--) {
29 | digit = parseInt(cardNumber.substring(i, i + 1), 10);
30 | if (timesTwo) {
31 | addend = digit * 2;
32 | if (addend > 9) {
33 | addend -= 9;
34 | }
35 | } else {
36 | addend = digit;
37 | }
38 | sum += addend;
39 | timesTwo = !timesTwo;
40 | }
41 |
42 | var modulus = sum % 10;
43 | if (modulus !== 0) {
44 | return false;
45 | }
46 |
47 | return true;
48 | };
49 |
50 | /*
51 | * Validate Card Holder Name. Returns true if card holder valid. Only allows
52 | * non-empty ISO/IEC 8859-1 values 100 characters or less.
53 | */
54 | var validateCardHolderName = function(cardHolderName) {
55 | // test for undefined
56 | if (!cardHolderName) {
57 | return false;
58 | }
59 |
60 | // test white space only
61 | if (!cardHolderName.trim()) {
62 | return false;
63 | }
64 |
65 | // test ISO/IEC 8859-1 characters between 1 and 100
66 | if (!/^[\u0020-\u007E\u00A0-\u00FF]{1,100}$/.test(cardHolderName)) {
67 | return false;
68 | }
69 |
70 | return true;
71 | };
72 |
73 | /*
74 | * Validate CVN. Applies to non-Amex card types. Only allows 3 numeric
75 | * characters.
76 | */
77 | var validateCvn = function(cvn) {
78 | // test numeric length 3
79 | if (!/^\d{3}$/.test(cvn)) {
80 | return false;
81 | }
82 |
83 | return true;
84 | };
85 |
86 | /*
87 | * Validate Amex CVN. Applies to Amex card types. Only allows 4 numeric
88 | * characters.
89 | */
90 | var validateAmexCvn = function(cvn) {
91 | // test numeric length 4
92 | if (!/^\d{4}$/.test(cvn)) {
93 | return false;
94 | }
95 |
96 | return true;
97 | };
98 |
99 | /*
100 | * Validate Expiry Date Format. Only allows 4 numeric characters. Month must
101 | * be between 1 and 12.
102 | */
103 | var validateExpiryDateFormat = function(expiryDate) {
104 |
105 | // test numeric of length 4
106 | if (!/^\d{4}$/.test(expiryDate)) {
107 | return false;
108 | }
109 |
110 | var month = parseInt(expiryDate.substring(0, 2), 10);
111 | var year = parseInt(expiryDate.substring(2, 4), 10);
112 |
113 | // test month range is 1-12
114 | if (month < 1 || month > 12) {
115 | return false;
116 | }
117 |
118 | return true;
119 | };
120 |
121 | /*
122 | * Validate Expiry Date Not In Past. Also runs checks from
123 | * validateExpiryDateFormat.
124 | */
125 | var validateExpiryDateNotInPast = function(expiryDate) {
126 | // test valid format
127 | if (!validateExpiryDateFormat(expiryDate)) {
128 | return false;
129 | }
130 |
131 | var month = parseInt(expiryDate.substring(0, 2), 10);
132 | var year = parseInt(expiryDate.substring(2, 4), 10);
133 |
134 | // test date is not in the past
135 | var now = new Date();
136 | var currentMonth = now.getMonth() + 1;
137 | var currentYear = now.getFullYear();
138 | if (year < (currentYear % 100)) {
139 | return false;
140 | } else if (year === (currentYear % 100) && month < currentMonth) {
141 | return false;
142 | }
143 |
144 | return true;
145 | };
146 |
147 | return {
148 | validateCardNumber : validateCardNumber,
149 | validateCardHolderName : validateCardHolderName,
150 | validateCvn : validateCvn,
151 | validateAmexCvn : validateAmexCvn,
152 | validateExpiryDateFormat : validateExpiryDateFormat,
153 | validateExpiryDateNotInPast : validateExpiryDateNotInPast
154 | };
155 | }());
156 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "rxp-js",
3 | "description": "The official Realex Payments JS Library",
4 | "version": "1.5.5",
5 | "homepage": "https://github.com/realexpayments/rxp-js",
6 | "author": {
7 | "name": "Realex Developer",
8 | "url": "http://www.realexpayments.com"
9 | },
10 | "repository": {
11 | "type": "git",
12 | "url": "https://github.com/realexpayments/rxp-js"
13 | },
14 | "licenses": [
15 | {
16 | "type": "MIT",
17 | "url": "https://github.com/realexpayments/rxp-js/blob/master/LICENSE-MIT"
18 | }
19 | ],
20 | "main": "lib/rxp-js",
21 | "engines": {
22 | "node": ">= 0.10.0"
23 | },
24 | "devDependencies": {
25 | "chromedriver": "^2.31.0",
26 | "geckodriver": "^1.8.0",
27 | "grunt": "^1.0.1",
28 | "grunt-contrib-concat": "^1.0.1",
29 | "grunt-contrib-jasmine": "^1.1.0",
30 | "grunt-contrib-jshint": "^1.1.0",
31 | "grunt-contrib-uglify": "^3.0.1",
32 | "grunt-contrib-watch": "^1.0.0",
33 | "grunt-php": "^1.5.1",
34 | "intern": "^3.4.6"
35 | },
36 | "keywords": [],
37 | "scripts": {
38 | "grunt": "grunt"
39 | }
40 | }
41 |
--------------------------------------------------------------------------------
/specs/functional/hpp/embedded-positives_spec.js:
--------------------------------------------------------------------------------
1 | define(function (require) {
2 | var bdd = require('intern!bdd');
3 | var assert = require('intern/chai!assert');
4 | var successHelper = require('intern/dojo/node!../../helpers/hpp').iframeSuccessHelper;
5 |
6 | bdd.describe('RealexRemote - HPP Embedded Positive Tests', function () {
7 | bdd.it('should process a payment successfully',
8 | successHelper(
9 | // url
10 | require.toUrl('http://localhost:8989/examples/hpp/process-a-payment-embedded.html'),
11 | // iframe selector
12 | '#targetIframe',
13 | // fields to enter
14 | [
15 | { name: 'pas_ccnum', type: 'text', value: '4111111111111111' },
16 | { name: 'pas_expiry', type: 'text', value: '1225' },
17 | { name: 'pas_cccvc', type: 'text', value: '012' },
18 | { name: 'pas_ccname', type: 'text', value: 'Jane Doe' },
19 | ],
20 | // callback to assert against result
21 | function (command) {
22 | return command
23 | .execute(() => document.body.innerText)
24 | .then(function (text) {
25 | // make our assertions on the HPP response
26 | var json = JSON.parse(text);
27 | json = JSON.parse(json.response);
28 | assert.isOk(json.AUTHCODE);
29 | })
30 | .end();
31 | }
32 | ).bind(this)
33 | );
34 | });
35 | });
36 |
--------------------------------------------------------------------------------
/specs/functional/hpp/lightbox-positives_spec.js:
--------------------------------------------------------------------------------
1 | define(function (require) {
2 | var bdd = require('intern!bdd');
3 | var assert = require('intern/chai!assert');
4 | var successHelper = require('intern/dojo/node!../../helpers/hpp').iframeSuccessHelper;
5 |
6 | bdd.describe('RealexRemote - HPP Lightbox Positive Tests', function () {
7 | bdd.it('should process a payment successfully',
8 | successHelper(
9 | // url
10 | require.toUrl('http://localhost:8989/examples/hpp/process-a-payment-lightbox.html'),
11 | // iframe selector
12 | '[id^="rxp-frame-"]',
13 | // fields to enter
14 | [
15 | { name: 'pas_ccnum', type: 'text', value: '4111111111111111' },
16 | { name: 'pas_expiry', type: 'text', value: '1225' },
17 | { name: 'pas_cccvc', type: 'text', value: '012' },
18 | { name: 'pas_ccname', type: 'text', value: 'Jane Doe' },
19 | ],
20 | // callback to assert against result
21 | function (command) {
22 | return command
23 | .execute(() => document.body.innerText)
24 | .then(function (text) {
25 | // make our assertions on the HPP response
26 | var json = JSON.parse(text);
27 | json = JSON.parse(json.response);
28 | assert.isOk(json.AUTHCODE);
29 | })
30 | .end();
31 | }
32 | ).bind(this)
33 | );
34 | });
35 | });
36 |
--------------------------------------------------------------------------------
/specs/functional/hpp/redirect-positives_spec.js:
--------------------------------------------------------------------------------
1 | define(function (require) {
2 | var bdd = require('intern!bdd');
3 | var assert = require('intern/chai!assert');
4 | var successHelper = require('intern/dojo/node!../../helpers/hpp').redirectSuccessHelper;
5 |
6 | bdd.describe('RealexRemote - HPP Redirect Positive Tests', function () {
7 | bdd.it('should process a payment successfully',
8 | successHelper(
9 | // url
10 | require.toUrl('http://localhost:8989/examples/hpp/redirect-for-payment.html'),
11 | // fields to enter
12 | [
13 | { name: 'pas_ccnum', type: 'text', value: '4111111111111111' },
14 | { name: 'pas_expiry', type: 'text', value: '1225' },
15 | { name: 'pas_cccvc', type: 'text', value: '012' },
16 | { name: 'pas_ccname', type: 'text', value: 'Jane Doe' },
17 | ],
18 | // callback to assert against result
19 | function (command) {
20 | return command
21 | .execute(() => document.body.innerText)
22 | .then(function (text) {
23 | // make our assertions on the HPP response
24 | assert.isOk(text.indexOf('Your transaction has been successful') !== -1);
25 | })
26 | .end();
27 | }
28 | ).bind(this)
29 | );
30 | });
31 | });
32 |
--------------------------------------------------------------------------------
/specs/helpers/hpp.js:
--------------------------------------------------------------------------------
1 | /**
2 | * Sets the current command/session to the desired URL.
3 | *
4 | * Require's the desired URL to add a `loaded` class to the
5 | * body on load complete.
6 | *
7 | * @param {leadfoot/Command} command
8 | * @param {string} url
9 | */
10 | function loadUrlAndWait(command, url) {
11 | return command
12 | // navigate to our test page
13 | .get(url)
14 | .setFindTimeout(5000)
15 | // wait until the HPP request producer finishes
16 | .findByCssSelector('body.loaded');
17 | }
18 |
19 | /**
20 | * Sets a page's field base on the defined field type.
21 | *
22 | * @param {leadfoot/Command} command
23 | * @param {object} field
24 | */
25 | function setField(command, field) {
26 | switch (field.type) {
27 | case 'text':
28 | default:
29 | return command
30 | .findById(field.name)
31 | .click()
32 | .type(field.value)
33 | .end();
34 | }
35 | }
36 |
37 | /**
38 | * Set's the given fields. Will call `callback`
39 | * with the command/session to complete any assertions.
40 | *
41 | * @param {leadfoot/Command} command
42 | * @param {object[]} fields
43 | * @param {Function} callback
44 | */
45 | function setFieldsAndSubmit(command, fields, callback) {
46 | // start - enter form data
47 | for (var i = 0; i < fields.length; i++) {
48 | command = setField(command, fields[i]);
49 | }
50 | // end - enter form data
51 |
52 | command = command
53 | // submit HPP
54 | .findById('rxp-primary-btn')
55 | .click()
56 | .end()
57 | // wait for redirect to HPP response consumer
58 | // TODO: figure out a way to do this without `sleep`
59 | .sleep(1500);
60 | return callback(command);
61 | }
62 |
63 | /**
64 | * Focuses a frame and set's the given fields. Will call `callback`
65 | * with the command/session to complete any assertions.
66 | *
67 | * @param {leadfoot/Command} command
68 | * @param {object[]} fields
69 | * @param {Function} callback
70 | */
71 | function setFrameFieldsAndSubmit(command, fields, callback) {
72 | return function (iframe) {
73 | // focus to the iframe
74 | command = command.switchToFrame(iframe);
75 |
76 | // start - enter form data
77 | for (var i = 0; i < fields.length; i++) {
78 | command = setField(command, fields[i]);
79 | }
80 | // end - enter form data
81 |
82 | command = command
83 | // submit HPP
84 | .findById('rxp-primary-btn')
85 | .click()
86 | .end()
87 | // wait for redirect to HPP response consumer
88 | // TODO: figure out a way to do this without `sleep`
89 | .sleep(1500)
90 | // ensure we're targeting the parent and not a non-existing iframe
91 | .switchToParentFrame();
92 | return callback(command);
93 | };
94 | }
95 |
96 | /**
97 | * Completes an HPP lightbox with the given fields. Will call `callback`
98 | * with the command/session to complete any assertions.
99 | *
100 | * @param {string} url
101 | * @param {object[]} fields
102 | * @param {Function} callback
103 | */
104 | function iframeSuccessHelper(url, iframeSelector, fields, callback) {
105 | return function () {
106 | var command = this.remote;
107 | return loadUrlAndWait(command, url)
108 | // start HPP
109 | .findById('payButtonId')
110 | .click()
111 | .end()
112 | // find the first iframe with an id that starts with our identifier
113 | .findByCssSelector(iframeSelector)
114 | .then(setFrameFieldsAndSubmit(command, fields, callback))
115 | .end();
116 | };
117 | }
118 |
119 | /**
120 | * Completes an HPP redirect with the given fields. Will call `callback`
121 | * with the command/session to complete any assertions.
122 | *
123 | * @param {string} url
124 | * @param {object[]} fields
125 | * @param {Function} callback
126 | */
127 | function redirectSuccessHelper(url, fields, callback) {
128 | return function () {
129 | var command = this.remote;
130 | return loadUrlAndWait(command, url)
131 | // start HPP
132 | .findById('payButtonId')
133 | .click()
134 | .end()
135 | .then(() => setFieldsAndSubmit(command, fields, callback));
136 | };
137 | }
138 |
139 | module.exports = {
140 | iframeSuccessHelper: iframeSuccessHelper,
141 | redirectSuccessHelper: redirectSuccessHelper,
142 | };
143 |
--------------------------------------------------------------------------------
/specs/intern.config.js:
--------------------------------------------------------------------------------
1 | // Learn more about configuring this file at .
2 | // These default settings work OK for most people. The options that *must* be changed below are the
3 | // packages, suites, excludeInstrumentation, and (if you want functional tests) functionalSuites.
4 | define({
5 | // Browsers to run integration testing against. Options that will be permutated are browserName, version, platform,
6 | // and platformVersion; any other capabilities options specified for an environment will be copied as-is. Note that
7 | // browser and platform names, and version number formats, may differ between cloud testing systems.
8 | environments: [
9 | {
10 | browserName: 'chrome',
11 | // Run headless when possible
12 | // Comment below line when using `leaveRemoteOpen: true`
13 | chromeOptions: { args: ['headless', 'disable-gpu'], }, fixSessionCapabilities: false,
14 | },
15 | ],
16 |
17 | // Uncomment to keep browser automation session open to inspect results of tests
18 | // leaveRemoteOpen: true,
19 |
20 | // Name of the tunnel class to use for WebDriver tests.
21 | // See for built-in options
22 | tunnel: 'SeleniumTunnel',
23 |
24 | // Unit test suite(s) to run in each browser
25 | functionalSuites: [ 'specs/functional/**/*_spec.js' ],
26 |
27 | // A regular expression matching URLs to files that should not be included in code coverage analysis. Set to `true`
28 | // to completely disable code coverage.
29 | excludeInstrumentation: true
30 | });
31 |
--------------------------------------------------------------------------------
/specs/unit/rxp-hpp_spec.js:
--------------------------------------------------------------------------------
1 | /*
2 | * Unit tests for rxp-hpp.js
3 | */
4 | describe('rxp-hpp library', function () {
5 | /*
6 | * Unit tests for createFormHiddenInput
7 | */
8 | describe('form input creation (createFormHiddenInput)', function () {
9 | it('creates element', function () {
10 | var field = RealexHpp._internal.createFormHiddenInput("name", "value");
11 | expect(field).not.toBe(null);
12 | expect(field.name).toBe("name");
13 | expect(field.value).toBe("value");
14 | });
15 |
16 | it('empty name', function () {
17 | var field = RealexHpp._internal.createFormHiddenInput("", "value");
18 | expect(field).not.toBe(null);
19 | expect(field.name).toBe("");
20 | expect(field.value).toBe("value");
21 | });
22 |
23 | it('empty value', function () {
24 | var field = RealexHpp._internal.createFormHiddenInput("name", "");
25 | expect(field).not.toBe(null);
26 | expect(field.name).toBe("name");
27 | expect(field.value).toBe("");
28 | });
29 |
30 | it('empty name and value', function () {
31 | var field = RealexHpp._internal.createFormHiddenInput("", "");
32 | expect(field).not.toBe(null);
33 | expect(field.name).toBe("");
34 | expect(field.value).toBe("");
35 | });
36 | });
37 |
38 | /*
39 | * Unit tests for checkDevicesOrientation
40 | */
41 | describe('device orientation (checkDevicesOrientation)', function () {
42 | it('0', function () {
43 | window.orientation = 0;
44 | var orientation = RealexHpp._internal.checkDevicesOrientation();
45 | expect(orientation).toBe(false);
46 | });
47 |
48 | it('90', function () {
49 | window.orientation = 90;
50 | var orientation = RealexHpp._internal.checkDevicesOrientation();
51 | expect(orientation).toBe(true);
52 | });
53 |
54 | it('180', function () {
55 | window.orientation = 180;
56 | var orientation = RealexHpp._internal.checkDevicesOrientation();
57 | expect(orientation).toBe(false);
58 | });
59 |
60 | it('-90', function () {
61 | window.orientation = -90;
62 | var orientation = RealexHpp._internal.checkDevicesOrientation();
63 | expect(orientation).toBe(true);
64 | });
65 | });
66 |
67 | /*
68 | * Unit tests for createOverlay
69 | */
70 | describe('lightbox overlay (createOverlay)', function () {
71 | it('creates an overlay', function () {
72 | var overlay = RealexHpp._internal.createOverlay();
73 | expect(overlay).not.toBe(null);
74 | expect(overlay.getAttribute('id')).toMatch(/rxp\-overlay\-/);
75 |
76 | var injectedOverlay = document.getElementById(overlay.getAttribute('id'));
77 | expect(overlay).toBe(injectedOverlay);
78 | });
79 | });
80 |
81 | /*
82 | * Unit tests for createCloseButton
83 | */
84 | describe('lightbox overlay close button (createCloseButton)', function () {
85 | it('creates an overlay', function () {
86 | var overlay = RealexHpp._internal.createOverlay();
87 | expect(overlay).not.toBe(null);
88 |
89 | var close = RealexHpp._internal.createCloseButton(overlay);
90 | expect(close).not.toBe(null);
91 | expect(close.getAttribute('id')).toMatch(/rxp\-frame\-close\-/);
92 | });
93 | });
94 |
95 | /*
96 | * Unit tests for createForm
97 | */
98 | describe('request form(createForm)', function () {
99 | it('creates form with no extra data', function () {
100 | var form = RealexHpp._internal.createForm(document, {});
101 | expect(form).not.toBe(null);
102 | expect(form.children.length).toBe(3);
103 | });
104 |
105 | it('creates redirect form with no extra data', function () {
106 | var form = RealexHpp._internal.createForm(document, {}, true);
107 | expect(form).not.toBe(null);
108 | expect(form.children.length).toBe(2);
109 | });
110 |
111 | it('creates form with extra data', function () {
112 | var form = RealexHpp._internal.createForm(document, {NAME: 'value'});
113 | expect(form).not.toBe(null);
114 | expect(form.children.length).toBe(4);
115 | });
116 |
117 | it('creates redirect form with extra data', function () {
118 | var form = RealexHpp._internal.createForm(document, {NAME: 'value'}, true);
119 | expect(form).not.toBe(null);
120 | expect(form.children.length).toBe(3);
121 | });
122 | });
123 |
124 | /*
125 | * Unit tests for createSpinner
126 | */
127 | describe('lightbox load indicator (createSpinner)', function () {
128 | it('creates an image', function () {
129 | var spinner = RealexHpp._internal.createSpinner();
130 | expect(spinner).not.toBe(null);
131 | expect(spinner.getAttribute('id')).toMatch(/rxp\-loader\-/);
132 | });
133 | });
134 |
135 | /*
136 | * Unit tests for getUrlParser
137 | */
138 | describe('url parsing (getUrlParser)', function () {
139 | var testUrl = 'http://hostname.com/path?query=true';
140 |
141 | it('parses url', function () {
142 | var url = RealexHpp._internal.getUrlParser(testUrl);
143 | expect(url).not.toBe(null);
144 | expect(url.hostname).toBe('hostname.com');
145 | expect(url.pathname).toBe('/path');
146 | });
147 | });
148 |
149 | /*
150 | * Unit tests for getHostnameFromUrl
151 | */
152 | describe('url parsing (getHostnameFromUrl)', function () {
153 | var testUrl = 'http://hostname.com/path?query=true';
154 |
155 | it('parses hostname', function () {
156 | var host = RealexHpp._internal.getHostnameFromUrl(testUrl);
157 | expect(host).toBe('hostname.com');
158 | });
159 | });
160 |
161 |
162 | /*
163 | * Unit tests for getHostnameFromUrl
164 | */
165 | describe('url parsing (getHostnameFromUrl)', function () {
166 | var testUrl = 'http://hostname.com/path?query=true';
167 |
168 | it('same host returns true', function () {
169 | var host = RealexHpp._internal.isMessageFromHpp(testUrl, testUrl);
170 | expect(host).toBe(true);
171 | });
172 |
173 | it('different hosts return false', function () {
174 | var host = RealexHpp._internal.isMessageFromHpp(testUrl, '#');
175 | expect(host).toBe(false);
176 | });
177 | });
178 | });
179 |
--------------------------------------------------------------------------------
/specs/unit/rxp-remote_spec.js:
--------------------------------------------------------------------------------
1 | /*
2 | * Unit tests for rxp-remote.js
3 | */
4 | describe( 'rxp-remote library', function () {
5 |
6 | /*
7 | * Unit tests for validateCardNumber
8 | */
9 | describe( 'card validation (validateCardNumber)', function () {
10 | it('valid card', function () {
11 | expect(RealexRemote.validateCardNumber('424242424242424242')).toBe(true);
12 | });
13 |
14 | it('non-numeric card', function () {
15 | expect(RealexRemote.validateCardNumber('a24242424242424242')).toBe(false);
16 | });
17 |
18 | it('card with spaces', function () {
19 | expect(RealexRemote.validateCardNumber('4242 424242424242')).toBe(false);
20 | });
21 |
22 | it('empty card', function () {
23 | expect(RealexRemote.validateCardNumber('')).toBe(false);
24 | });
25 |
26 | it('undefined card', function () {
27 | expect(RealexRemote.validateCardNumber()).toBe(false);
28 | });
29 |
30 | it('white space only', function () {
31 | expect(RealexRemote.validateCardNumber(' ')).toBe(false);
32 | });
33 |
34 | it('length < 12', function () {
35 | expect(RealexRemote.validateCardNumber('42424242420')).toBe(false);
36 | });
37 |
38 | it('length > 19', function () {
39 | expect(RealexRemote.validateCardNumber('42424242424242424242')).toBe(false);
40 | });
41 |
42 | it('length = 12', function () {
43 | expect(RealexRemote.validateCardNumber('424242424242')).toBe(true);
44 | });
45 |
46 | it('length = 19', function () {
47 | expect(RealexRemote.validateCardNumber('4242424242424242428')).toBe(true);
48 | });
49 |
50 | it('luhn check', function () {
51 | expect(RealexRemote.validateCardNumber('4242424242424242427')).toBe(false);
52 | });
53 | });
54 |
55 | /*
56 | * Unit tests for validateCardHolderName
57 | */
58 | describe( 'card holder name validation (validateCardHolderName)', function () {
59 | it('valid name', function () {
60 | expect(RealexRemote.validateCardHolderName('Joe Smith')).toBe(true);
61 | });
62 |
63 | it('empty name', function () {
64 | expect(RealexRemote.validateCardHolderName('')).toBe(false);
65 | });
66 |
67 | it('undefined name', function () {
68 | expect(RealexRemote.validateCardHolderName()).toBe(false);
69 | });
70 |
71 | it('white space only', function () {
72 | expect(RealexRemote.validateCardHolderName(' ')).toBe(false);
73 | });
74 |
75 | it('name of 100 characters', function () {
76 | expect(RealexRemote.validateCardHolderName('abcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghij')).toBe(true);
77 | });
78 |
79 | it('name over 100 characters', function () {
80 | expect(RealexRemote.validateCardHolderName('abcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghija')).toBe(false);
81 | });
82 |
83 | it('ISO/IEC 8859-1 characters 1', function () {
84 | expect(RealexRemote.validateCardHolderName('!\" # $ % & \' ( ) * + - . / 0 1 2 3 4 5 6 7 8 9 : ; < = > ? @ A B C D E F G H I J K L M N O P Q R')).toBe(true);
85 | });
86 |
87 | it('ISO/IEC 8859-1 characters 2', function () {
88 | expect(RealexRemote.validateCardHolderName('S T U V W X Y Z [ \ ] ^ _ ` a b c d e f g h i j k l m n o p q r s t u v w x y z { | } ~ ¡ ¢ £ ¤ ¥')).toBe(true);
89 | });
90 |
91 | it('ISO/IEC 8859-1 characters 3', function () {
92 | expect(RealexRemote.validateCardHolderName('¦ § ¨ © ª « ¬ ® ¯ ° ± ² ³ ´ µ ¶ · ¸ ¹ º » ¼ ½ ¾ ¿ À Á Â Ã Ä Å Æ Ç È É Ê Ë Ì Í Î Ï Ð Ñ Ò Ó Ô Õ Ö')).toBe(true);
93 | });
94 |
95 | it('ISO/IEC 8859-1 characters 4', function () {
96 | expect(RealexRemote.validateCardHolderName('× Ø Ù Ú Û Ü Ý Þ ß à á â ã ä å æ ç è é ê ë ì í î ï ð ñ ò ó ô õ ö ÷ ø ù ú û ü ý þ ÿ')).toBe(true);
97 | });
98 |
99 | it('non-ISO/IEC 8859-1 characters', function () {
100 | expect(RealexRemote.validateCardHolderName('€')).toBe(false);
101 | });
102 |
103 | });
104 |
105 | /*
106 | * Unit tests for validateAmexCvn
107 | */
108 | describe( 'CVN Amex validation (validateAmexCvn)', function () {
109 | it('valid Amex CVN', function () {
110 | expect(RealexRemote.validateAmexCvn('1234')).toBe(true);
111 | });
112 |
113 | it('empty CVN', function () {
114 | expect(RealexRemote.validateAmexCvn('')).toBe(false);
115 | });
116 |
117 | it('undefined CVN', function () {
118 | expect(RealexRemote.validateAmexCvn()).toBe(false);
119 | });
120 |
121 | it('white space only', function () {
122 | expect(RealexRemote.validateAmexCvn(' ')).toBe(false);
123 | });
124 |
125 | it('Amex CVN of 5 numbers', function () {
126 | expect(RealexRemote.validateAmexCvn('12345')).toBe(false);
127 | });
128 |
129 | it('Amex CVN of 3 numbers', function () {
130 | expect(RealexRemote.validateAmexCvn('123')).toBe(false);
131 | });
132 |
133 | it('non-numeric Amex CVN of 4 characters', function () {
134 | expect(RealexRemote.validateAmexCvn('123a')).toBe(false);
135 | });
136 |
137 | });
138 |
139 | /*
140 | * Unit tests for validateCvn
141 | */
142 | describe( 'CVN non-Amex validation (validateCvn)', function () {
143 | it('valid non-Amex CVN', function () {
144 | expect(RealexRemote.validateCvn('123')).toBe(true);
145 | });
146 |
147 | it('empty CVN', function () {
148 | expect(RealexRemote.validateCvn('')).toBe(false);
149 | });
150 |
151 | it('undefined CVN', function () {
152 | expect(RealexRemote.validateCvn()).toBe(false);
153 | });
154 |
155 | it('white space only', function () {
156 | expect(RealexRemote.validateCvn(' ')).toBe(false);
157 | });
158 |
159 | it('non-Amex CVN of 4 numbers', function () {
160 | expect(RealexRemote.validateCvn('1234')).toBe(false);
161 | });
162 |
163 | it('non-Amex CVN of 2 numbers', function () {
164 | expect(RealexRemote.validateCvn('12')).toBe(false);
165 | });
166 |
167 | it('non-numeric non-Amex CVN of 3 characters', function () {
168 | expect(RealexRemote.validateCvn('12a')).toBe(false);
169 | });
170 | });
171 |
172 | /*
173 | * Unit tests for validateExpiryDateFormat
174 | */
175 | describe( 'Expiry date format validation (validateExpiryDateFormat)', function () {
176 |
177 | it('valid date 1299', function () {
178 | expect(RealexRemote.validateExpiryDateFormat('1299')).toBe(true);
179 | });
180 |
181 | it('valid date 0199', function () {
182 | expect(RealexRemote.validateExpiryDateFormat('0199')).toBe(true);
183 | });
184 |
185 | it('non-numeric date', function () {
186 | expect(RealexRemote.validateExpiryDateFormat('a199')).toBe(false);
187 | });
188 |
189 | it('date with spaces', function () {
190 | expect(RealexRemote.validateExpiryDateFormat('1 99')).toBe(false);
191 | });
192 |
193 | it('empty date', function () {
194 | expect(RealexRemote.validateExpiryDateFormat('')).toBe(false);
195 | });
196 |
197 | it('undefined date', function () {
198 | expect(RealexRemote.validateExpiryDateFormat()).toBe(false);
199 | });
200 |
201 | it('white space only', function () {
202 | expect(RealexRemote.validateExpiryDateFormat(' ')).toBe(false);
203 | });
204 |
205 | it('length > 4', function () {
206 | expect(RealexRemote.validateExpiryDateFormat('12099')).toBe(false);
207 | });
208 |
209 | it('length < 4', function () {
210 | expect(RealexRemote.validateExpiryDateFormat('199')).toBe(false);
211 | });
212 |
213 | it('invalid month 00', function () {
214 | expect(RealexRemote.validateExpiryDateFormat('0099')).toBe(false);
215 | });
216 |
217 | it('invalid month 13', function () {
218 | expect(RealexRemote.validateExpiryDateFormat('1399')).toBe(false);
219 | });
220 | });
221 |
222 | /*
223 | * Unit tests for validateExpiryDateNotInPast
224 | */
225 | describe( 'Expiry date not in past validation (validateExpiryDateNotInPast)', function () {
226 |
227 | it('date in past', function () {
228 | expect(RealexRemote.validateExpiryDateNotInPast('0615')).toBe(false);
229 | });
230 |
231 | it('current month', function () {
232 | var now = new Date();
233 | var nowMonth = '' + (now.getMonth() + 1);
234 | nowMonth = nowMonth.length < 2 ? '0' + nowMonth : nowMonth;
235 | var nowYear = ('' + now.getFullYear()).substr(2,4);
236 | expect(RealexRemote.validateExpiryDateNotInPast(nowMonth + nowYear)).toBe(true);
237 | });
238 | });
239 |
240 | });
--------------------------------------------------------------------------------