globalpayments / rxp-js

Global Payments Ecommerce JavaScript Library
MIT License
34 stars 70 forks source link

200 Comms error handling #16

Open blackn1ght opened 6 years ago

blackn1ght commented 6 years ago

When you use the test card for the 200 - Comms Error (Visa, 4009830000001985), the rxp-js library throws an error on the receiveMessage function.

The issue is on line 323: if (event.data && JSON.parse(event.data).iframe) {

The value of event.data is

Error: 200
Message: [ test system ] COMMS ERROR

but the code is expecting it to be a JSON.

Arnaud-Paytweak commented 6 years ago

Exactly the same problem here when the payment as already be done and refused. If the customer retry he will be stuck on an black screen with no information.

The javascript don't support Error messages and the customer has only the close button on a black shadow....

For information we get this JSON after the ajax auth :

{"status":"501","message":"Error: 501
Message: This transaction (PTWKTEST1805181600) has already been processed! If you feel this is incorrect please contact the merchant!","errors":[],"data":{"response":{"timestamp":"20180518163641","targetClientId":null,"targetUserName":null,"userResultCode":"0","userDefaultMessage":null,"acquirertransactionid":null,"merchantid":null,"account":null,"orderid":"PTWKTEST1805181600","authcode":null,"result":"501","cvnresult":null,"avspostcoderesponse":null,"avsaddressresponse":null,"batchid":null,"message":"This transaction (PTWKTEST1805181600) has already been processed! If you feel this is incorrect please contact the merchant!","errmsg":null, ...

Opa- commented 6 years ago

Same here on 5xx & 2xx messages. Any update on this ?

violetann commented 5 years ago

it also breaks if the ORDER_ID is the same as one already processed it sends the error message back in the event.data as a string not a json string so it also breaks there too.

rodrigobb commented 5 years ago

Same here, any updates on this?

Is the same error here https://github.com/globalpayments/rxp-js/issues/22, which makes the lightbox unusable.

carlbradwell17 commented 4 years ago

I have also ran into this problem. I made a quick modification to the script to alert any response which is text rather than JSON. There is a function "testJSON" and an extra if...else statement starting on line 307, that's it:

testJSON

`/jslint browser:true / var RealexHpp = (function () {

'use strict';

var hppUrl = "https://pay.realexpayments.com/pay";

var randomId = randomId || Math.random().toString(16).substr(2,8);

var setHppUrl = function(url) {
    hppUrl = url;
};

var isWindowsMobileOs = /Windows Phone|IEMobile/.test(navigator.userAgent);
var isAndroidOrIOs = /Android|iPad|iPhone|iPod/.test(navigator.userAgent);
var isMobileXS =  ( (((window.innerWidth > 0) ? window.innerWidth : screen.width) <= 360 ? true : false) || (((window.innerHeight > 0) ? window.innerHeight : screen.Height) <= 360 ? true : false)) ;

// Display IFrame on WIndows Phone OS mobile devices
var isMobileIFrame = isWindowsMobileOs;

// For IOs/Android and small screen devices always open in new tab/window
var isMobileNewTab = !isWindowsMobileOs && (isAndroidOrIOs || isMobileXS);
var tabWindow;

var redirectUrl;

var internal = {
    createFormHiddenInput: function (name, value) {
        var el = document.createElement("input");
        el.setAttribute("type", "hidden");
        el.setAttribute("name", name);
        el.setAttribute("value", value);
        return el;
    },

    checkDevicesOrientation: function () {
        if (window.orientation === 90 || window.orientation === -90) {
            return true;
        } else {
            return false;
        }
    },

    createOverlay: function () {
        var overlay = document.createElement("div");
        overlay.setAttribute("id", "rxp-overlay-" + randomId);
        overlay.style.position = "fixed";
        overlay.style.width = "100%";
        overlay.style.height = "100%";
        overlay.style.top = "0";
        overlay.style.left = "0";
        overlay.style.transition = "all 0.3s ease-in-out";
        overlay.style.zIndex = "100";

        if (isMobileIFrame) {
            overlay.style.position = "absolute !important";
            overlay.style.WebkitOverflowScrolling = "touch";
            overlay.style.overflowX = "hidden";
            overlay.style.overflowY = "scroll";
        }

        document.body.appendChild(overlay);

        setTimeout(function () {
            overlay.style.background = "rgba(0, 0, 0, 0.7)";
        }, 1);

        return overlay;
    },

    closeModal: function (closeButton, iFrame, spinner, overlayElement) {
        if (closeButton && closeButton.parentNode) {
            closeButton.parentNode.removeChild(closeButton);
        }

        if (iFrame && iFrame.parentNode) {
            iFrame.parentNode.removeChild(iFrame);
        }

        if (spinner && spinner.parentNode) {
            spinner.parentNode.removeChild(spinner);
        }

        if (!overlayElement) {
            return;
        }

        overlayElement.className = "";
        setTimeout(function () {
            if (overlayElement.parentNode) {
                overlayElement.parentNode.removeChild(overlayElement);
            }
        }, 300);
    },

    createCloseButton: function (overlayElement) {
        if (document.getElementById("rxp-frame-close-" + randomId) !== null) {
            return;
        }

        var closeButton = document.createElement("img");
        closeButton.setAttribute("id","rxp-frame-close-" + randomId);
        closeButton.setAttribute("src", "");
        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;");

        setTimeout(function () {
            closeButton.style.opacity = "1";
        },500);

        if (isMobileIFrame) {
            closeButton.style.position = "absolute";
            closeButton.style.float = "right";
            closeButton.style.top = "20px";
            closeButton.style.left = "initial";
            closeButton.style.marginLeft = "0px";
            closeButton.style.right = "20px";
        }

        return closeButton;
    },

    createForm: function (doc, token, ignorePostMessage) {
        var form = document.createElement("form");
        form.setAttribute("method", "POST");
        form.setAttribute("action", hppUrl);

        var versionSet = false;

        for (var key in token) {
            if (key === "HPP_VERSION"){
                versionSet = true;
            }
            form.appendChild(internal.createFormHiddenInput(key, token[key]));
        }

        if (versionSet === false){
            form.appendChild(internal.createFormHiddenInput("HPP_VERSION", "2"));
        }

        if (ignorePostMessage) {
            form.appendChild(internal.createFormHiddenInput("MERCHANT_RESPONSE_URL", redirectUrl));
        } else {
            var parser = internal.getUrlParser(window.location.href);
            var hppOriginParam = parser.protocol + '//' + parser.host;

            form.appendChild(internal.createFormHiddenInput("HPP_POST_RESPONSE", hppOriginParam));
            form.appendChild(internal.createFormHiddenInput("HPP_POST_DIMENSIONS", hppOriginParam));
        }
        return form;
    },

    createSpinner: function () {
        var spinner = document.createElement("img");
        spinner.setAttribute("src", "");
        spinner.setAttribute("id", "rxp-loader-" + randomId);
        spinner.style.left = "50%";
        spinner.style.position = "fixed";
        spinner.style.background = "#FFFFFF";
        spinner.style.borderRadius = "50%";
        spinner.style.width = "30px";
        spinner.style.zIndex = "200";
        spinner.style.marginLeft = "-15px";
        spinner.style.top = "120px";
        return spinner;
    },

    createIFrame: function (overlayElement, token) {
        //Create the spinner
        var spinner = internal.createSpinner();
        document.body.appendChild(spinner);

        //Create the iframe
        var iFrame = document.createElement("iframe");
        iFrame.setAttribute("name", "rxp-frame-" + randomId);
        iFrame.setAttribute("id", "rxp-frame-" + randomId);
        iFrame.setAttribute("height", "562px");
        iFrame.setAttribute("frameBorder", "0");
        iFrame.setAttribute("width", "360px");
        iFrame.setAttribute("seamless", "seamless");

        iFrame.style.zIndex = "10001";
        iFrame.style.position = "absolute";
        iFrame.style.transition = "transform 0.5s ease-in-out";
        iFrame.style.transform = "scale(0.7)";
        iFrame.style.opacity = "0";

        overlayElement.appendChild(iFrame);

        if (isMobileIFrame) {
            iFrame.style.top = "0px";
            iFrame.style.bottom = "0px";
            iFrame.style.left = "0px";
            iFrame.style.marginLeft = "0px;";
            iFrame.style.width = "100%";
            iFrame.style.height = "100%";
            iFrame.style.minHeight = "100%";
            iFrame.style.WebkitTransform = "translate3d(0,0,0)";
            iFrame.style.transform = "translate3d(0, 0, 0)";

            var metaTag = document.createElement('meta');
            metaTag.name = "viewport";
            metaTag.content = "width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=0";
            document.getElementsByTagName('head')[0].appendChild(metaTag);
        } else {
            iFrame.style.top = "40px";
            iFrame.style.left = "50%";
            iFrame.style.marginLeft = "-180px";
        }

        var closeButton;

        iFrame.onload = function () {
            iFrame.style.opacity = "1";
            iFrame.style.transform = "scale(1)";
            iFrame.style.backgroundColor = "#ffffff";

            if (spinner.parentNode) {
                spinner.parentNode.removeChild(spinner);
            }

            closeButton = internal.createCloseButton();
            overlayElement.appendChild(closeButton);
            closeButton.addEventListener("click", function () {
                internal.closeModal(closeButton, iFrame, spinner, overlayElement);
            }, true);
        };

        var form = internal.createForm(document, token);
        if (iFrame.contentWindow.document.body) {
            iFrame.contentWindow.document.body.appendChild(form);
        } else {
            iFrame.contentWindow.document.appendChild(form);
        }

        form.submit();

        return {
            spinner: spinner,
            iFrame: iFrame,
            closeButton: closeButton
        };
    },

    openWindow: function (token) {
        //open new window
        var tabWindow = window.open();

        // browsers can prevent a new window from being created
        // e.g. mobile Safari
        if (!tabWindow) {
            return null;
        }

        var doc = tabWindow.document;

        //add meta tag to new window (needed for iOS 8 bug)
        var meta = doc.createElement("meta");
        var name = doc.createAttribute("name");
        name.value = "viewport";
        meta.setAttributeNode(name);
        var content = doc.createAttribute("content");
        content.value = "width=device-width";
        meta.setAttributeNode(content);
        doc.head.appendChild(meta);

        //create form, append to new window and submit
        var form = internal.createForm(doc, token);
        doc.body.appendChild(form);
        form.submit();

        return tabWindow;
    },

    getUrlParser: function (url) {
        var parser = document.createElement('a');
        parser.href = url;
        return parser;
    },

    getHostnameFromUrl: function (url) {
  return internal.getUrlParser(url).hostname;
    },

    isMessageFromHpp: function (origin, hppUrl) {
        return internal.getHostnameFromUrl(origin) === internal.getHostnameFromUrl(hppUrl);
    },

    receiveMessage: function (lightboxInstance, merchantUrl, isEmbedded) {
        return function (event) {
            //Check the origin of the response comes from HPP
            if (!internal.isMessageFromHpp(event.origin, hppUrl)) {
                return;
            }

            function testJSON(text) { 
                if (typeof text !== "string") { 
                    return false; 
                } 
                try { 
                    JSON.parse(text); 
                    return true; 
                } catch (error) { 
                    return false; 
                } 
            }

            if (event.data && testJSON(event.data)) {

                // check for iframe resize values
                if (JSON.parse(event.data).iframe) {
                    if (!isMobileNewTab) {
                        var iframeWidth = JSON.parse(event.data).iframe.width;
                        var iframeHeight = JSON.parse(event.data).iframe.height;

                        var iFrame;
                        var resized = false;

                        if (isEmbedded) {
                            iFrame = lightboxInstance.getIframe();
                        } else {
                            iFrame = document.getElementById("rxp-frame-" + randomId);
                        }

                        if (iframeWidth === "390px" && iframeHeight === "440px") {
                            iFrame.setAttribute("width", iframeWidth);
                            iFrame.setAttribute("height", iframeHeight);
                            resized = true;
                        }

                        iFrame.style.backgroundColor="#ffffff";

                        if (isMobileIFrame) {
                            iFrame.style.marginLeft = "0px";
                            iFrame.style.WebkitOverflowScrolling = "touch";
                            iFrame.style.overflowX = "scroll";
                            iFrame.style.overflowY = "scroll";

                            if (!isEmbedded) {
                                var overlay = document.getElementById("rxp-overlay-" + randomId);
                                overlay.style.overflowX = "scroll";
                                overlay.style.overflowY = "scroll";
                            }
                        } else if (!isEmbedded && resized) {
                            iFrame.style.marginLeft = (parseInt(iframeWidth.replace("px", ""), 10) / 2 * -1) + "px";
                        }

                        if (!isEmbedded && resized) {
                            // wrap the below in a setTimeout to prevent a timing issue on a
                            // cache-miss load
                            setTimeout(function () {
                                var closeButton = document.getElementById("rxp-frame-close-" + randomId);
                                closeButton.style.marginLeft = ((parseInt(iframeWidth.replace("px", ""), 10) / 2) -7) + "px";
                            }, 200);
                        }
                    }
                } else {
                    if (isMobileNewTab && tabWindow) {
                        //Close the new window
                        tabWindow.close();
                    } else {
                        //Close the lightbox
                        lightboxInstance.close();
                    }

                    var response = event.data;

                    //Create a form and submit the hpp response to the merchant's response url
                    var form = document.createElement("form");
                    form.setAttribute("method", "POST");
                    form.setAttribute("action", merchantUrl);

                    form.appendChild(internal.createFormHiddenInput("hppResponse", response));

                    document.body.appendChild(form);

                    form.submit();
                }
            }
            else {

                alert(event.data);

            }
        };
    }
};

// Initialising some variables used throughout this file.
var RxpLightbox = (function () {
    var instance;

    function init() {
        var overlayElement;
        var spinner;
        var iFrame;
        var closeButton;
        var token;
        var isLandscape = internal.checkDevicesOrientation();

        if (isMobileIFrame) {
            if (window.addEventListener) {
                    window.addEventListener("orientationchange", function () {
                    isLandscape = internal.checkDevicesOrientation();
                }, false);
            }
        }

        return {
            lightbox: function () {
                if (isMobileNewTab) {
                    tabWindow = internal.openWindow(token);
                } else {
                    overlayElement = internal.createOverlay();
                    var temp = internal.createIFrame(overlayElement, token);
                    spinner = temp.spinner;
                    iFrame = temp.iFrame;
                    closeButton = temp.closeButton;
                }
            },
            close: function () {
                internal.closeModal();
            },
            setToken: function (hppToken) {
                token = hppToken;
            }
        };
    }

    return {
        // Get the Singleton instance if one exists
        // or create one if it doesn't
        getInstance: function (hppToken) {
            if (!instance) {
                instance = init();
            }

            //Set the hpp token
            instance.setToken(hppToken);

            return instance;
        },
        init: function (idOfLightboxButton, merchantUrl, serverSdkJson) {
            //Get the lightbox instance (it's a singleton) and set the sdk json
            var lightboxInstance = RxpLightbox.getInstance(serverSdkJson);

            // Sets the event listener on the PAY button. The click will invoke the lightbox method
            if (document.getElementById(idOfLightboxButton).addEventListener) {
                document.getElementById(idOfLightboxButton).addEventListener("click", lightboxInstance.lightbox, true);
            } else {
                document.getElementById(idOfLightboxButton).attachEvent('onclick', lightboxInstance.lightbox);
            }

            if (window.addEventListener) {
                window.addEventListener("message", internal.receiveMessage(lightboxInstance, merchantUrl), false);
            } else {
                window.attachEvent('message', internal.receiveMessage(lightboxInstance, merchantUrl));
            }
        }
    };
})();

// Initialising some variables used throughout this file.
var RxpEmbedded = (function () {
    var instance;

    function init() {
        var overlayElement;
        var spinner;
        var iFrame;
        var closeButton;
        var token;

        return {
            embedded: function () {
                var form = internal.createForm(document, token);
                if (iFrame) {
                    if (iFrame.contentWindow.document.body) {
                        iFrame.contentWindow.document.body.appendChild(form);
                    } else {
                        iFrame.contentWindow.document.appendChild(form);
                    }
                    form.submit();
                    iFrame.style.display = "inherit";
                }
            },
            close: function () {
                iFrame.style.display = "none";
            },
            setToken: function (hppToken) {
                token = hppToken;
            },
            setIframe: function (iframeId) {
                iFrame = document.getElementById(iframeId);
            },
            getIframe: function () {
                return iFrame;
            }
        };
    }

    return {
        // Get the Singleton instance if one exists
        // or create one if it doesn't
        getInstance: function (hppToken) {
            if (!instance) {
                instance = init();
            }

            //Set the hpp token
            instance.setToken(hppToken);

            return instance;
        },
        init: function (idOfEmbeddedButton, idOfTargetIframe, merchantUrl, serverSdkJson) {
            //Get the embedded instance (it's a singleton) and set the sdk json
            var embeddedInstance = RxpEmbedded.getInstance(serverSdkJson);

            embeddedInstance.setIframe(idOfTargetIframe);

            // Sets the event listener on the PAY button. The click will invoke the embedded method
            if (document.getElementById(idOfEmbeddedButton).addEventListener) {
                document.getElementById(idOfEmbeddedButton).addEventListener("click", embeddedInstance.embedded, true);
            } else {
                document.getElementById(idOfEmbeddedButton).attachEvent('onclick', embeddedInstance.embedded);
            }

            if (window.addEventListener) {
                window.addEventListener("message", internal.receiveMessage(embeddedInstance, merchantUrl, true), false);
            } else {
                window.attachEvent('message', internal.receiveMessage(embeddedInstance, merchantUrl, true));
            }
        }
    };
})();

var RxpRedirect = (function () {
    var instance;

    function init() {
        var overlayElement;
        var spinner;
        var iFrame;
        var closeButton;
        var token;
        var isLandscape = internal.checkDevicesOrientation();

        if (isMobileIFrame) {
            if (window.addEventListener) {
                    window.addEventListener("orientationchange", function () {
                    isLandscape = internal.checkDevicesOrientation();
                }, false);
            }
        }

        return {
            redirect: function () {
                var form = internal.createForm(document, token, true);
                document.body.append(form);
                form.submit();
            },
            setToken: function (hppToken) {
                token = hppToken;
            }
        };
    }
    return {
        // Get the singleton instance if one exists
        // or create one if it doesn't
        getInstance: function (hppToken) {
            if (!instance) {
                instance = init();
            }

            // Set the hpp token
            instance.setToken(hppToken);

            return instance;
        },
        init: function (idOfButton, merchantUrl, serverSdkJson) {
            // Get the redirect instance (it's a singleton) and set the sdk json
            var redirectInstance = RxpRedirect.getInstance(serverSdkJson);
            redirectUrl = merchantUrl;

            // Sets the event listener on the PAY button. The click will invoke the redirect method
            if (document.getElementById(idOfButton).addEventListener) {
                document.getElementById(idOfButton).addEventListener("click", redirectInstance.redirect, true);
            } else {
                document.getElementById(idOfButton).attachEvent('onclick', redirectInstance.redirect);
            }

            if (window.addEventListener) {
                window.addEventListener("message", internal.receiveMessage(redirectInstance, merchantUrl), false);
            } else {
                window.attachEvent('message', internal.receiveMessage(redirectInstance, merchantUrl));
            }
        }
    };
}());

// RealexHpp
return {
    init: RxpLightbox.init,
    lightbox: {
        init: RxpLightbox.init
    },
    embedded: {
        init: RxpEmbedded.init
    },
    redirect: {
        init: RxpRedirect.init
    },
    setHppUrl: setHppUrl,
    _internal: internal
};

}());`

mattcoles commented 2 years ago

Any news on this? The error used to be returned JSON format but recently it has contained a HTML string.

This blows up when using responseJson = Request.Form[0]; in C# with the .Net Sdk when parsing the response. We are now having to catch this and use responseJson = Request.Unvalidated.Form[0]; which is not ideal!