From a4630ca8537a514f261e3efcd79192b1ca28fae8 Mon Sep 17 00:00:00 2001 From: Shane Logsdon Date: Tue, 31 Aug 2021 09:45:40 -0400 Subject: [PATCH 1/4] add a local response viewer --- .../process-a-payment-embedded-autoload.html | 2 +- examples/hpp/process-a-payment-embedded.html | 2 +- examples/hpp/process-a-payment-lightbox.html | 2 +- examples/hpp/response.php | 44 +++++++++++++++++++ 4 files changed, 47 insertions(+), 3 deletions(-) create mode 100644 examples/hpp/response.php diff --git a/examples/hpp/process-a-payment-embedded-autoload.html b/examples/hpp/process-a-payment-embedded-autoload.html index d695c7d..5d6965b 100644 --- a/examples/hpp/process-a-payment-embedded-autoload.html +++ b/examples/hpp/process-a-payment-embedded-autoload.html @@ -13,7 +13,7 @@ RealexHpp.embedded.init( "autoload", "targetIframe", - "https://dev.rlxcarts.com/mobileSDKsV2/response.php", + "/examples/hpp/response.php", jsonFromServerSdk, { onResize:function(data){ diff --git a/examples/hpp/process-a-payment-embedded.html b/examples/hpp/process-a-payment-embedded.html index a82e96a..0878ccd 100644 --- a/examples/hpp/process-a-payment-embedded.html +++ b/examples/hpp/process-a-payment-embedded.html @@ -13,7 +13,7 @@ RealexHpp.embedded.init( "payButtonId", "targetIframe", - "https://dev.rlxcarts.com/mobileSDKsV2/response.php", // merchant url + "/examples/hpp/response.php", // merchant url jsonFromServerSdk, // form data { // options onResize:function(data){ diff --git a/examples/hpp/process-a-payment-lightbox.html b/examples/hpp/process-a-payment-lightbox.html index 90fe77b..cca827f 100644 --- a/examples/hpp/process-a-payment-lightbox.html +++ b/examples/hpp/process-a-payment-lightbox.html @@ -12,7 +12,7 @@ $.getJSON("/examples/hpp/proxy-request.php?slug=process-a-payment", function (jsonFromServerSdk) { RealexHpp.lightbox.init( "payButtonId", - "https://dev.rlxcarts.com/mobileSDKsV2/response.php", // merchant url + "/examples/hpp/response.php", // merchant url jsonFromServerSdk //form data ); $('body').addClass('loaded'); diff --git a/examples/hpp/response.php b/examples/hpp/response.php new file mode 100644 index 0000000..ec289ec --- /dev/null +++ b/examples/hpp/response.php @@ -0,0 +1,44 @@ + $v) { + try { + $hppResponse[$k] = base64_decode($v); + } catch (Exception $e) { + /* */ + } + } +} +catch (Exception $e) { + $hppResponse = $originalHppResponse; +} + +?> + + + HPP Demo Response + + + +

HPP Demo Response

+
+
+
+
+ Try Again +
+ + From af81424f48bdaeba1dfa2020f6a7b30dedd6d03b Mon Sep 17 00:00:00 2001 From: Shane Logsdon Date: Tue, 7 Sep 2021 22:55:19 -0400 Subject: [PATCH 2/4] perform basic check that data is from hpp HPP should only send the parent window string messages, so if a non-string value is sent (e.g. from a browser extension), we ignore it. --- lib/rxp-hpp.js | 25 ++++++++++++++++++++----- 1 file changed, 20 insertions(+), 5 deletions(-) diff --git a/lib/rxp-hpp.js b/lib/rxp-hpp.js index 192ff75..ef3f065 100644 --- a/lib/rxp-hpp.js +++ b/lib/rxp-hpp.js @@ -159,6 +159,10 @@ var RealexHpp = (function () { var _r; + if (typeof answer !== "string") { + return null; + } + try { _r=JSON.parse(answer); } catch (e) { @@ -442,9 +446,21 @@ var RealexHpp = (function () { if (!internal.isMessageFromHpp(e.event.origin, hppUrl)) { return; } + + if (!e.event.data) { + return; + } + + var evtdata = internal.decodeAnswer(e.event.data); + + // we received an invalid message from the HPP iframe (e.g. from a browser plugin) + // return early to prevent invalid processing + if (evtdata === null) { + return; + } + // check for iframe resize values - var evtdata; - if (e.event.data && (evtdata=internal.decodeAnswer(e.event.data)).iframe) { + if (evtdata.iframe) { if (!isMobileNewTab()) { var iframeWidth = evtdata.iframe.width; var iframeHeight = evtdata.iframe.height; @@ -510,9 +526,8 @@ var RealexHpp = (function () { }; var response = e.event.data; //allow the script to intercept the answer, instead of redirecting to another page. (which is really a 90s thing) - if(typeof e.url==='function'){ - var answer=internal.decodeAnswer(response); - e.url(answer,_close); + if (typeof e.url === 'function'){ + e.url(evtdata, _close); return; } _close(); From 6a0a47622d548344f586a8c69daee01b43d04c44 Mon Sep 17 00:00:00 2001 From: Shane Logsdon Date: Tue, 7 Sep 2021 23:27:13 -0400 Subject: [PATCH 3/4] add basic code documentation --- lib/rxp-hpp.js | 144 +++++++++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 140 insertions(+), 4 deletions(-) diff --git a/lib/rxp-hpp.js b/lib/rxp-hpp.js index ef3f065..428fe77 100644 --- a/lib/rxp-hpp.js +++ b/lib/rxp-hpp.js @@ -45,8 +45,16 @@ var RealexHpp = (function () { var redirectUrl; + /** + * Shared functionality across lightbox, embedded, and redirect display modes. + */ var internal = { evtMsg: [], + /** + * Adds a new window message event listener and tracks it for later removal + * + * @param {Function} evtMsgFct + */ addEvtMsgListener: function(evtMsgFct) { this.evtMsg.push({ fct: evtMsgFct, opt: false }); if (window.addEventListener) { @@ -55,6 +63,9 @@ var RealexHpp = (function () { window.attachEvent('message', evtMsgFct); } }, + /** + * Removes a previously set window message event listener + */ removeOldEvtMsgListener: function () { if (this.evtMsg.length > 0) { var evt = this.evtMsg.pop(); @@ -65,6 +76,9 @@ var RealexHpp = (function () { } } }, + /** + * Shimmed base64 encode/decode support + */ base64:{ encode:function(input) { var keyStr = "ABCDEFGHIJKLMNOP" + @@ -155,6 +169,18 @@ var RealexHpp = (function () { return unescape(output); } }, + /** + * Converts an HPP message to a developer-friendly version. + * + * The decode process has two steps: + * + * 1. Attempt to parse the string as JSON. If this fails, an error response + * is provided as we expect that the HPP has errored out to the cardholder + * 2. Attempt to base64 decode the data to cover both HPP versions 1 and 2. + * + * @param {any} answer + * @returns null if answer is not a string, otherwise the data from the HPP + */ decodeAnswer:function(answer){ //internal.decodeAnswer var _r; @@ -178,6 +204,13 @@ var RealexHpp = (function () { } catch (e) { /** */ } return _r; }, + /** + * Creates a new input of type `hidden`. Does not append to DOM. + * + * @param {string} name Name for the new input + * @param {string} value Value for the new input + * @returns the created input + */ createFormHiddenInput: function (name, value) { var el = document.createElement("input"); el.setAttribute("type", "hidden"); @@ -186,6 +219,11 @@ var RealexHpp = (function () { return el; }, + /** + * Determines a mobile device's orientation for width calculation + * + * @returns true if in landscape + */ checkDevicesOrientation: function () { if (window.orientation === 90 || window.orientation === -90) { return true; @@ -194,6 +232,12 @@ var RealexHpp = (function () { } }, + /** + * Creates a semi-transparent overlay with full width/height to serve as + * a background for the lightbox modal + * + * @returns the created overlay + */ createOverlay: function () { var overlay = document.createElement("div"); overlay.setAttribute("id", "rxp-overlay-" + randomId); @@ -221,6 +265,14 @@ var RealexHpp = (function () { return overlay; }, + /** + * Closes a lightbox modal and all associated elements + * + * @param {HTMLImageElement} closeButton + * @param {HTMLIFrameElement} iFrame + * @param {HTMLImageElement} spinner + * @param {HTMLDivElement} overlayElement + */ closeModal: function (closeButton, iFrame, spinner, overlayElement) { if (closeButton && closeButton.parentNode) { closeButton.parentNode.removeChild(closeButton); @@ -246,6 +298,11 @@ var RealexHpp = (function () { }, 300); }, + /** + * Creates a close button for the lightbox modal + * + * @returns the created element + */ createCloseButton: function (overlayElement) { if (document.getElementById("rxp-frame-close-" + randomId) !== null) { return; @@ -272,6 +329,19 @@ var RealexHpp = (function () { return closeButton; }, + /** + * Creates a form and appends the HPP request data as hidden input elements to + * POST to the defined HPP URL. + * + * The created form is not appended to the DOM and is not submitted at this time. + * + * @param {Document} doc + * @param {object} token HPP request data + * @param {bool} ignorePostMessage If true, the HPP will redirect to the defined + * defined redirect URL. If false, the HPP will send a postMessage + * to the parent window to be handled by this library. + * @returns the created form + */ createForm: function (doc, token, ignorePostMessage) { var form = document.createElement("form"); form.setAttribute("method", "POST"); @@ -302,6 +372,12 @@ var RealexHpp = (function () { return form; }, + /** + * Creates a visual spinner element to be shown with the lightbox overlay while the + * HPP's iframe loads + * + * @returns the created spinner element + */ createSpinner: function () { var spinner = document.createElement("img"); spinner.setAttribute("src", ""); @@ -317,6 +393,14 @@ var RealexHpp = (function () { return spinner; }, + /** + * Creates the HPP's form, spinner, iframe, and close button, appends them + * to the DOM, and submits the form to load the HPP + * + * @param {HTMLDivElement} overlayElement + * @param {object} token The HPP request data + * @returns an object with the created spinner, iframe, and close button + */ createIFrame: function (overlayElement, token) { //Create the spinner var spinner = internal.createSpinner(); @@ -397,6 +481,18 @@ var RealexHpp = (function () { }; }, + /** + * Opens the HPP in a new window + * + * Used in some mobile scenarios or when the browser viewport is + * smaller than the HPP's inner width. + * + * Will automatically post the request data to the defined HPP + * URL to load the HPP. + * + * @param {object} token The HPP request data + * @returns the created window + */ openWindow: function (token) { //open new window var tabWindow = window.open(); @@ -427,20 +523,51 @@ var RealexHpp = (function () { return tabWindow; }, + /** + * Creates a rudimentary URL parser using an anchor element + * + * @param {string} url + * @returns the created anchor element + */ getUrlParser: function (url) { var parser = document.createElement('a'); parser.href = url; return parser; }, + /** + * Gets the hostname/origin from a URL. Used for origin checks + * + * @param {string} url + * @returns the hostname/origin of the URL + */ getHostnameFromUrl: function (url) { - return internal.getUrlParser(url).hostname; + return internal.getUrlParser(url).hostname; }, + /** + * Compares the origins from both arguments to validate we have received a postMessage + * from the expected source + * + * @param {string} origin The origin attached to the recieved message + * @param {string} hppUrl Our expected source origin + * @returns true if the origins match + */ isMessageFromHpp: function (origin, hppUrl) { return internal.getHostnameFromUrl(origin) === internal.getHostnameFromUrl(hppUrl); }, + /** + * Handles messages from the HPP + * + * Messages from the HPP are one of: + * + * - iframe resize event + * - transaction response + * - error information + * + * @param {MessageEvent} e + */ receiveMessage: function (e) { //Check the origin of the response comes from HPP if (!internal.isMessageFromHpp(e.event.origin, hppUrl)) { @@ -542,7 +669,9 @@ var RealexHpp = (function () { } }; - // Initialising some variables used throughout this file. + /** + * Public interface for the lightbox display mode + */ var RxpLightbox = (function () { var instance; @@ -621,7 +750,9 @@ var RealexHpp = (function () { }; })(); - // Initialising some variables used throughout this file. + /** + * Public interface for the embedded display mode + */ var RxpEmbedded = (function () { var instance; @@ -700,6 +831,9 @@ var RealexHpp = (function () { }; })(); + /** + * Public interface for the redirect display mode + */ var RxpRedirect = (function () { var instance; @@ -766,7 +900,9 @@ var RealexHpp = (function () { }; }()); - // RealexHpp + /** + * Public interface for the Realex HPP library + */ return { init: RxpLightbox.init, lightbox: { From 511c5ff8e6ea3242ff8f9b1e9bca6b2ed41f970c Mon Sep 17 00:00:00 2001 From: Shane Logsdon Date: Tue, 7 Sep 2021 23:27:38 -0400 Subject: [PATCH 4/4] increment the version --- dist/rxp-js.js | 171 ++++++++++++++++++++++++++++++++++++++++++--- dist/rxp-js.min.js | 4 +- package.json | 2 +- 3 files changed, 164 insertions(+), 13 deletions(-) diff --git a/dist/rxp-js.js b/dist/rxp-js.js index 6b7b1ef..ba1ada7 100644 --- a/dist/rxp-js.js +++ b/dist/rxp-js.js @@ -1,4 +1,4 @@ -/*! rxp-js - v1.5.0 - 2021-07-27 +/*! rxp-js - v1.5.1 - 2021-09-07 * The official Realex Payments JS Library * https://github.com/realexpayments/rxp-js * Licensed MIT @@ -49,8 +49,16 @@ var RealexHpp = (function () { var redirectUrl; + /** + * Shared functionality across lightbox, embedded, and redirect display modes. + */ var internal = { evtMsg: [], + /** + * Adds a new window message event listener and tracks it for later removal + * + * @param {Function} evtMsgFct + */ addEvtMsgListener: function(evtMsgFct) { this.evtMsg.push({ fct: evtMsgFct, opt: false }); if (window.addEventListener) { @@ -59,6 +67,9 @@ var RealexHpp = (function () { window.attachEvent('message', evtMsgFct); } }, + /** + * Removes a previously set window message event listener + */ removeOldEvtMsgListener: function () { if (this.evtMsg.length > 0) { var evt = this.evtMsg.pop(); @@ -69,6 +80,9 @@ var RealexHpp = (function () { } } }, + /** + * Shimmed base64 encode/decode support + */ base64:{ encode:function(input) { var keyStr = "ABCDEFGHIJKLMNOP" + @@ -159,10 +173,26 @@ var RealexHpp = (function () { return unescape(output); } }, + /** + * Converts an HPP message to a developer-friendly version. + * + * The decode process has two steps: + * + * 1. Attempt to parse the string as JSON. If this fails, an error response + * is provided as we expect that the HPP has errored out to the cardholder + * 2. Attempt to base64 decode the data to cover both HPP versions 1 and 2. + * + * @param {any} answer + * @returns null if answer is not a string, otherwise the data from the HPP + */ decodeAnswer:function(answer){ //internal.decodeAnswer var _r; + if (typeof answer !== "string") { + return null; + } + try { _r=JSON.parse(answer); } catch (e) { @@ -178,6 +208,13 @@ var RealexHpp = (function () { } catch (e) { /** */ } return _r; }, + /** + * Creates a new input of type `hidden`. Does not append to DOM. + * + * @param {string} name Name for the new input + * @param {string} value Value for the new input + * @returns the created input + */ createFormHiddenInput: function (name, value) { var el = document.createElement("input"); el.setAttribute("type", "hidden"); @@ -186,6 +223,11 @@ var RealexHpp = (function () { return el; }, + /** + * Determines a mobile device's orientation for width calculation + * + * @returns true if in landscape + */ checkDevicesOrientation: function () { if (window.orientation === 90 || window.orientation === -90) { return true; @@ -194,6 +236,12 @@ var RealexHpp = (function () { } }, + /** + * Creates a semi-transparent overlay with full width/height to serve as + * a background for the lightbox modal + * + * @returns the created overlay + */ createOverlay: function () { var overlay = document.createElement("div"); overlay.setAttribute("id", "rxp-overlay-" + randomId); @@ -221,6 +269,14 @@ var RealexHpp = (function () { return overlay; }, + /** + * Closes a lightbox modal and all associated elements + * + * @param {HTMLImageElement} closeButton + * @param {HTMLIFrameElement} iFrame + * @param {HTMLImageElement} spinner + * @param {HTMLDivElement} overlayElement + */ closeModal: function (closeButton, iFrame, spinner, overlayElement) { if (closeButton && closeButton.parentNode) { closeButton.parentNode.removeChild(closeButton); @@ -246,6 +302,11 @@ var RealexHpp = (function () { }, 300); }, + /** + * Creates a close button for the lightbox modal + * + * @returns the created element + */ createCloseButton: function (overlayElement) { if (document.getElementById("rxp-frame-close-" + randomId) !== null) { return; @@ -272,6 +333,19 @@ var RealexHpp = (function () { return closeButton; }, + /** + * Creates a form and appends the HPP request data as hidden input elements to + * POST to the defined HPP URL. + * + * The created form is not appended to the DOM and is not submitted at this time. + * + * @param {Document} doc + * @param {object} token HPP request data + * @param {bool} ignorePostMessage If true, the HPP will redirect to the defined + * defined redirect URL. If false, the HPP will send a postMessage + * to the parent window to be handled by this library. + * @returns the created form + */ createForm: function (doc, token, ignorePostMessage) { var form = document.createElement("form"); form.setAttribute("method", "POST"); @@ -302,6 +376,12 @@ var RealexHpp = (function () { return form; }, + /** + * Creates a visual spinner element to be shown with the lightbox overlay while the + * HPP's iframe loads + * + * @returns the created spinner element + */ createSpinner: function () { var spinner = document.createElement("img"); spinner.setAttribute("src", ""); @@ -317,6 +397,14 @@ var RealexHpp = (function () { return spinner; }, + /** + * Creates the HPP's form, spinner, iframe, and close button, appends them + * to the DOM, and submits the form to load the HPP + * + * @param {HTMLDivElement} overlayElement + * @param {object} token The HPP request data + * @returns an object with the created spinner, iframe, and close button + */ createIFrame: function (overlayElement, token) { //Create the spinner var spinner = internal.createSpinner(); @@ -397,6 +485,18 @@ var RealexHpp = (function () { }; }, + /** + * Opens the HPP in a new window + * + * Used in some mobile scenarios or when the browser viewport is + * smaller than the HPP's inner width. + * + * Will automatically post the request data to the defined HPP + * URL to load the HPP. + * + * @param {object} token The HPP request data + * @returns the created window + */ openWindow: function (token) { //open new window var tabWindow = window.open(); @@ -427,28 +527,71 @@ var RealexHpp = (function () { return tabWindow; }, + /** + * Creates a rudimentary URL parser using an anchor element + * + * @param {string} url + * @returns the created anchor element + */ getUrlParser: function (url) { var parser = document.createElement('a'); parser.href = url; return parser; }, + /** + * Gets the hostname/origin from a URL. Used for origin checks + * + * @param {string} url + * @returns the hostname/origin of the URL + */ getHostnameFromUrl: function (url) { - return internal.getUrlParser(url).hostname; + return internal.getUrlParser(url).hostname; }, + /** + * Compares the origins from both arguments to validate we have received a postMessage + * from the expected source + * + * @param {string} origin The origin attached to the recieved message + * @param {string} hppUrl Our expected source origin + * @returns true if the origins match + */ isMessageFromHpp: function (origin, hppUrl) { return internal.getHostnameFromUrl(origin) === internal.getHostnameFromUrl(hppUrl); }, + /** + * Handles messages from the HPP + * + * Messages from the HPP are one of: + * + * - iframe resize event + * - transaction response + * - error information + * + * @param {MessageEvent} e + */ receiveMessage: function (e) { //Check the origin of the response comes from HPP if (!internal.isMessageFromHpp(e.event.origin, hppUrl)) { return; } + + if (!e.event.data) { + return; + } + + var evtdata = internal.decodeAnswer(e.event.data); + + // we received an invalid message from the HPP iframe (e.g. from a browser plugin) + // return early to prevent invalid processing + if (evtdata === null) { + return; + } + // check for iframe resize values - var evtdata; - if (e.event.data && (evtdata=internal.decodeAnswer(e.event.data)).iframe) { + if (evtdata.iframe) { if (!isMobileNewTab()) { var iframeWidth = evtdata.iframe.width; var iframeHeight = evtdata.iframe.height; @@ -514,9 +657,8 @@ var RealexHpp = (function () { }; var response = e.event.data; //allow the script to intercept the answer, instead of redirecting to another page. (which is really a 90s thing) - if(typeof e.url==='function'){ - var answer=internal.decodeAnswer(response); - e.url(answer,_close); + if (typeof e.url === 'function'){ + e.url(evtdata, _close); return; } _close(); @@ -531,7 +673,9 @@ var RealexHpp = (function () { } }; - // Initialising some variables used throughout this file. + /** + * Public interface for the lightbox display mode + */ var RxpLightbox = (function () { var instance; @@ -610,7 +754,9 @@ var RealexHpp = (function () { }; })(); - // Initialising some variables used throughout this file. + /** + * Public interface for the embedded display mode + */ var RxpEmbedded = (function () { var instance; @@ -689,6 +835,9 @@ var RealexHpp = (function () { }; })(); + /** + * Public interface for the redirect display mode + */ var RxpRedirect = (function () { var instance; @@ -755,7 +904,9 @@ var RealexHpp = (function () { }; }()); - // RealexHpp + /** + * Public interface for the Realex HPP library + */ return { init: RxpLightbox.init, lightbox: { diff --git a/dist/rxp-js.min.js b/dist/rxp-js.min.js index 56ad758..f08dd15 100644 --- a/dist/rxp-js.min.js +++ b/dist/rxp-js.min.js @@ -1,7 +1,7 @@ -/*! rxp-js - v1.5.0 - 2021-07-27 +/*! rxp-js - v1.5.1 - 2021-09-07 * The official Realex Payments JS Library * https://github.com/realexpayments/rxp-js * Licensed MIT */ -Element.prototype.remove=function(){this.parentElement.removeChild(this)},NodeList.prototype.remove=HTMLCollection.prototype.remove=function(){for(var e=this.length-1;e>=0;e--)this[e]&&this[e].parentElement&&this[e].parentElement.removeChild(this[e])};var RealexHpp=function(){"use strict";var e,t,n="https://pay.realexpayments.com/pay",A=A||Math.random().toString(16).substr(2,8),i=360,o=/Windows Phone|IEMobile/.test(navigator.userAgent),r=/Android|iPad|iPhone|iPod/.test(navigator.userAgent),a=function(){return(window.innerWidth>0?window.innerWidth:screen.width)<=i||(window.innerHeight>0?window.innerHeight:screen.Height)<=i},d=o,s=function(){return!o&&(r||a())},c={evtMsg:[],addEvtMsgListener:function(e){this.evtMsg.push({fct:e,opt:!1}),window.addEventListener?window.addEventListener("message",e,!1):window.attachEvent("message",e)},removeOldEvtMsgListener:function(){if(this.evtMsg.length>0){var e=this.evtMsg.pop();window.addEventListener?window.removeEventListener("message",e.fct,e.opt):window.detachEvent("message",e.fct)}},base64:{encode:function(e){var t="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";e=escape(e);var n,A,i,o,r,a="",d="",s="",c=0;do{i=(n=e.charCodeAt(c++))>>2,o=(3&n)<<4|(A=e.charCodeAt(c++))>>4,r=(15&A)<<2|(d=e.charCodeAt(c++))>>6,s=63&d,isNaN(A)?r=s=64:isNaN(d)&&(s=64),a=a+t.charAt(i)+t.charAt(o)+t.charAt(r)+t.charAt(s),n=A=d="",i=o=r=s=""}while(c