UNPKG

@qooxdoo/framework

Version:

The JS Framework for Coders

1,065 lines (848 loc) 27.7 kB
/* ************************************************************************ qooxdoo - the new era of web development http://qooxdoo.org Copyright: 2004-2008 1&1 Internet AG, Germany, http://www.1und1.de 2006 Derrell Lipman 2006 STZ-IDA, Germany, http://www.stz-ida.de License: MIT: https://opensource.org/licenses/MIT See the LICENSE file in the project's top-level directory for details. Authors: * Sebastian Werner (wpbasti) * Andreas Ecker (ecker) * Derrell Lipman (derrell) * Andreas Junghans (lucidcake) ************************************************************************ */ /** * Transport layer to control which transport class (XmlHttp, Iframe or Script) * can be used. * * NOTE: Instances of this class must be disposed of after use * * @use(qx.io.remote.transport.Iframe) * @use(qx.io.remote.transport.Script) * @internal */ qx.Class.define("qx.io.remote.Exchange", { extend : qx.core.Object, implement : [ qx.core.IDisposable ], /* ***************************************************************************** CONSTRUCTOR ***************************************************************************** */ /** * Constructor method. * * @param vRequest {qx.io.remote.Request} request object */ construct : function(vRequest) { this.base(arguments); this.setRequest(vRequest); vRequest.setTransport(this); }, /* ***************************************************************************** EVENTS ***************************************************************************** */ events : { /** Fired whenever a request is send */ "sending" : "qx.event.type.Event", /** Fired whenever a request is received */ "receiving" : "qx.event.type.Event", /** Fired whenever a request is completed */ "completed" : "qx.io.remote.Response", /** Fired whenever a request is aborted */ "aborted" : "qx.event.type.Event", /** Fired whenever a request has failed */ "failed" : "qx.io.remote.Response", /** Fired whenever a request has timed out */ "timeout" : "qx.io.remote.Response" }, /* ***************************************************************************** STATICS ***************************************************************************** */ statics : { /* ************************************************************************ Class data, properties and methods ************************************************************************ */ /* --------------------------------------------------------------------------- TRANSPORT TYPE HANDLING --------------------------------------------------------------------------- */ /** * Predefined order of types. * * @internal */ typesOrder : [ "qx.io.remote.transport.XmlHttp", "qx.io.remote.transport.Iframe", "qx.io.remote.transport.Script" ], /** * Marker for initialized types. * * @internal */ typesReady : false, /** * Map of all available types. * * @internal */ typesAvailable : {}, /** * Map of all supported types. * * @internal */ typesSupported : {}, /** * Registers a transport type. * At the moment one out of XmlHttp, Iframe or Script. * * @param vClass {Object} transport class * @param vId {String} unique id */ registerType : function(vClass, vId) { qx.io.remote.Exchange.typesAvailable[vId] = vClass; }, /** * Initializes the available type of transport classes and * checks for the supported ones. * * @throws {Error} an error if no supported transport type is available */ initTypes : function() { if (qx.io.remote.Exchange.typesReady) { return; } for (var vId in qx.io.remote.Exchange.typesAvailable) { var vTransporterImpl = qx.io.remote.Exchange.typesAvailable[vId]; if (vTransporterImpl.isSupported()) { qx.io.remote.Exchange.typesSupported[vId] = vTransporterImpl; } } qx.io.remote.Exchange.typesReady = true; if (qx.lang.Object.isEmpty(qx.io.remote.Exchange.typesSupported)) { throw new Error("No supported transport types were found!"); } }, /** * Checks which supported transport class can handle the request with the * given content type. * * @param vImpl {Object} transport implementation * @param vNeeds {Map} requirements for the request like e.g. "cross-domain" * @param vResponseType {String} content type * @return {Boolean} <code>true</code> if the transport implementation supports * the request's requirements */ canHandle : function(vImpl, vNeeds, vResponseType) { if (!vImpl.handles.responseTypes.includes(vResponseType)) { return false; } for (var vKey in vNeeds) { if (!vImpl.handles[vKey]) { return false; } } return true; }, /* --------------------------------------------------------------------------- MAPPING --------------------------------------------------------------------------- */ /** * http://msdn.microsoft.com/en-us/library/ie/ms534359%28v=vs.85%29.aspx * * 0: UNINITIALIZED * The object has been created, but not initialized (the open method has not been called). * * 1: LOADING * The object has been created, but the send method has not been called. * * 2: LOADED * The send method has been called, but the status and headers are not yet available. * * 3: INTERACTIVE * Some data has been received. Calling the responseBody and responseText properties at this state to obtain partial results will return an error, because status and response headers are not fully available. * * 4: COMPLETED * All the data has been received, and the complete data is available in the * * @internal */ _nativeMap : { 0 : "created", 1 : "configured", 2 : "sending", 3 : "receiving", 4 : "completed" }, /* --------------------------------------------------------------------------- UTILS --------------------------------------------------------------------------- */ /** * Called from the transport class when a request was completed. * * @param vStatusCode {Integer} status code of the request * @param vReadyState {String} readystate of the request * @param vIsLocal {Boolean} whether the request is a local one * @return {Boolean | var} Returns boolean value depending on the status code */ wasSuccessful : function(vStatusCode, vReadyState, vIsLocal) { if (vIsLocal) { switch(vStatusCode) { case null: case 0: return true; case -1: // Not Available (OK for readystates: MSXML<4=1-3, MSXML>3=1-2, Gecko=1) return vReadyState < 4; default: // at least older versions of Safari don't set the status code for local file access return typeof vStatusCode === "undefined"; } } else { switch(vStatusCode) { case -1: // Not Available (OK for readystates: MSXML<4=1-3, MSXML>3=1-2, Gecko=1) if (qx.core.Environment.get("qx.debug")) { if (qx.core.Environment.get("qx.debug.io.remote") && vReadyState > 3) { qx.log.Logger.debug(this, "Failed with statuscode: -1 at readyState " + vReadyState); } } return vReadyState < 4; case 200: // OK case 304: // Not Modified return true; case 201: // Created case 202: // Accepted case 203: // Non-Authoritative Information case 204: // No Content case 205: // Reset Content return true; case 206: // Partial Content if (qx.core.Environment.get("qx.debug")) { if (qx.core.Environment.get("qx.debug.io.remote") && vReadyState === 4) { qx.log.Logger.debug(this, "Failed with statuscode: 206 (Partial content while being complete!)"); } } return vReadyState !== 4; case 300: // Multiple Choices case 301: // Moved Permanently case 302: // Moved Temporarily case 303: // See Other case 305: // Use Proxy case 400: // Bad Request case 401: // Unauthorized case 402: // Payment Required case 403: // Forbidden case 404: // Not Found case 405: // Method Not Allowed case 406: // Not Acceptable case 407: // Proxy Authentication Required case 408: // Request Time-Out case 409: // Conflict case 410: // Gone case 411: // Length Required case 412: // Precondition Failed case 413: // Request Entity Too Large case 414: // Request-URL Too Large case 415: // Unsupported Media Type case 500: // Server Error case 501: // Not Implemented case 502: // Bad Gateway case 503: // Out of Resources case 504: // Gateway Time-Out case 505: // HTTP Version not supported if (qx.core.Environment.get("qx.debug")) { if (qx.core.Environment.get("qx.debug.io.remote")) { qx.log.Logger.debug(this, "Failed with typical HTTP statuscode: " + vStatusCode); } } return false; // The following case labels are wininet.dll error codes that may // be encountered. // Server timeout case 12002: // Internet Name Not Resolved case 12007: // 12029 to 12031 correspond to dropped connections. case 12029: case 12030: case 12031: // Connection closed by server. case 12152: // See above comments for variable status. case 13030: if (qx.core.Environment.get("qx.debug")) { if (qx.core.Environment.get("qx.debug.io.remote")) { qx.log.Logger.debug(this, "Failed with MSHTML specific HTTP statuscode: " + vStatusCode); } } return false; default: // Handle all 20x status codes as OK as defined in the corresponding RFC // http://www.w3.org/Protocols/rfc2616/rfc2616.html if (vStatusCode > 206 && vStatusCode < 300) { return true; } qx.log.Logger.debug(this, "Unknown status code: " + vStatusCode + " (" + vReadyState + ")"); return false; } } }, /** * Status code to string conversion * * @param vStatusCode {Integer} request status code * @return {String} String presentation of status code */ statusCodeToString : function(vStatusCode) { switch(vStatusCode) { case -1: return "Not available"; case 0: // Attempt to generate a potentially meaningful error. // Get the current URL var url = window.location.href; // Are we on a local page obtained via file: protocol? if (url.toLowerCase().startsWith("file:")) { // Yup. Can't issue remote requests from here. return ("Unknown status code. " + "Possibly due to application URL using 'file:' protocol?"); } else { return ("Unknown status code. " + "Possibly due to a cross-domain request?"); } break; case 200: return "Ok"; case 304: return "Not modified"; case 206: return "Partial content"; case 204: return "No content"; case 300: return "Multiple choices"; case 301: return "Moved permanently"; case 302: return "Moved temporarily"; case 303: return "See other"; case 305: return "Use proxy"; case 400: return "Bad request"; case 401: return "Unauthorized"; case 402: return "Payment required"; case 403: return "Forbidden"; case 404: return "Not found"; case 405: return "Method not allowed"; case 406: return "Not acceptable"; case 407: return "Proxy authentication required"; case 408: return "Request time-out"; case 409: return "Conflict"; case 410: return "Gone"; case 411: return "Length required"; case 412: return "Precondition failed"; case 413: return "Request entity too large"; case 414: return "Request-URL too large"; case 415: return "Unsupported media type"; case 500: return "Server error"; case 501: return "Not implemented"; case 502: return "Bad gateway"; case 503: return "Out of resources"; case 504: return "Gateway time-out"; case 505: return "HTTP version not supported"; case 12002: return "Server timeout"; case 12029: return "Connection dropped"; case 12030: return "Connection dropped"; case 12031: return "Connection dropped"; case 12152: return "Connection closed by server"; case 13030: return "MSHTML-specific HTTP status code"; default: return "Unknown status code"; } } }, /* ***************************************************************************** PROPERTIES ***************************************************************************** */ properties : { /** Set the request to send with this transport. */ request : { check : "qx.io.remote.Request", nullable : true }, /** * Set the implementation to use to send the request with. * * The implementation should be a subclass of qx.io.remote.transport.Abstract and * must implement all methods in the transport API. */ implementation : { check : "qx.io.remote.transport.Abstract", nullable : true, apply : "_applyImplementation" }, /** Current state of the transport layer. */ state : { check : [ "configured", "sending", "receiving", "completed", "aborted", "timeout", "failed" ], init : "configured", event : "changeState", apply : "_applyState" } }, /* ***************************************************************************** MEMBERS ***************************************************************************** */ members : { /* --------------------------------------------------------------------------- CORE METHODS --------------------------------------------------------------------------- */ /** * Sends the request. * * @return {var | Boolean} Returns true if the request was sent. * @lint ignoreUnused(field) */ send : function() { var vRequest = this.getRequest(); if (!vRequest) { return this.error("Please attach a request object first"); } qx.io.remote.Exchange.initTypes(); var vUsage = qx.io.remote.Exchange.typesOrder; var vSupported = qx.io.remote.Exchange.typesSupported; // Mapping settings to contenttype and needs to check later // if the selected transport implementation can handle // fulfill these requirements. var vResponseType = vRequest.getResponseType(); var vNeeds = {}; if (vRequest.getAsynchronous()) { vNeeds.asynchronous = true; } else { vNeeds.synchronous = true; } if (vRequest.getCrossDomain()) { vNeeds.crossDomain = true; } if (vRequest.getFileUpload()) { vNeeds.fileUpload = true; } // See if there are any programmatic form fields requested for (var field in vRequest.getFormFields()) { // There are. vNeeds.programmaticFormFields = true; // No need to search further break; } var vTransportImpl, vTransport; for (var i=0, l=vUsage.length; i<l; i++) { vTransportImpl = vSupported[vUsage[i]]; if (vTransportImpl) { if (!qx.io.remote.Exchange.canHandle(vTransportImpl, vNeeds, vResponseType)) { continue; } try { if (qx.core.Environment.get("qx.debug")) { if (qx.core.Environment.get("qx.debug.io.remote")) { this.debug("Using implementation: " + vTransportImpl.classname); } } vTransport = new vTransportImpl; this.setImplementation(vTransport); vTransport.setUseBasicHttpAuth(vRequest.getUseBasicHttpAuth()); vTransport.send(); return true; } catch(ex) { this.error("Request handler throws error"); this.error(ex); return false; } } } this.error("There is no transport implementation available to handle this request: " + vRequest); }, /** * Force the transport into the aborted ("aborted") * state. * */ abort : function() { var vImplementation = this.getImplementation(); if (vImplementation) { if (qx.core.Environment.get("qx.debug")) { if (qx.core.Environment.get("qx.debug.io.remote")) { this.debug("Abort: implementation " + vImplementation.toHashCode()); } } vImplementation.abort(); } else { if (qx.core.Environment.get("qx.debug")) { if (qx.core.Environment.get("qx.debug.io.remote")) { this.debug("Abort: forcing state to be aborted"); } } this.setState("aborted"); } }, /** * Force the transport into the timeout state. * */ timeout : function() { var vImplementation = this.getImplementation(); if (vImplementation) { var str = ""; for (var key in vImplementation.getParameters()) { str += "&" + key + "=" + vImplementation.getParameters()[key]; } this.warn("Timeout: implementation " + vImplementation.toHashCode() + ", " + vImplementation.getUrl() + " [" + vImplementation.getMethod() + "], " + str); vImplementation.timeout(); } else { this.warn("Timeout: forcing state to timeout"); this.setState("timeout"); } // Disable future timeouts in case user handler blocks this.__disableRequestTimeout(); }, /* --------------------------------------------------------------------------- PRIVATES --------------------------------------------------------------------------- */ /** * Disables the timer of the request to prevent that the timer is expiring * even if the user handler (e.g. "completed") was already called. * */ __disableRequestTimeout : function() { var vRequest = this.getRequest(); if (vRequest) { vRequest.setTimeout(0); } }, /* --------------------------------------------------------------------------- EVENT HANDLER --------------------------------------------------------------------------- */ /** * Event listener for "sending" event. * * @param e {qx.event.type.Event} event object */ _onsending : function(e) { this.setState("sending"); }, /** * Event listener for "receiving" event. * * @param e {qx.event.type.Event} event object */ _onreceiving : function(e) { this.setState("receiving"); }, /** * Event listener for "completed" event. * * @param e {qx.event.type.Event} event object */ _oncompleted : function(e) { this.setState("completed"); }, /** * Event listener for "abort" event. * * @param e {qx.event.type.Event} event object */ _onabort : function(e) { this.setState("aborted"); }, /** * Event listener for "failed" event. * * @param e {qx.event.type.Event} event object */ _onfailed : function(e) { this.setState("failed"); }, /** * Event listener for "timeout" event. * * @param e {qx.event.type.Event} event object */ _ontimeout : function(e) { this.setState("timeout"); }, /* --------------------------------------------------------------------------- APPLY ROUTINES --------------------------------------------------------------------------- */ /** * Apply method for the implementation property. * * @param value {var} Current value * @param old {var} Previous value */ _applyImplementation : function(value, old) { if (old) { old.removeListener("sending", this._onsending, this); old.removeListener("receiving", this._onreceiving, this); old.removeListener("completed", this._oncompleted, this); old.removeListener("aborted", this._onabort, this); old.removeListener("timeout", this._ontimeout, this); old.removeListener("failed", this._onfailed, this); } if (value) { var vRequest = this.getRequest(); value.setUrl(vRequest.getUrl()); value.setMethod(vRequest.getMethod()); value.setAsynchronous(vRequest.getAsynchronous()); value.setUsername(vRequest.getUsername()); value.setPassword(vRequest.getPassword()); value.setParameters(vRequest.getParameters(false)); value.setFormFields(vRequest.getFormFields()); value.setRequestHeaders(vRequest.getRequestHeaders()); // Set the parseJson property which is currently only supported for XmlHttp transport // (which is the only transport supporting JSON parsing so far). if (value instanceof qx.io.remote.transport.XmlHttp){ value.setParseJson(vRequest.getParseJson()); } var data = vRequest.getData(); if (data === null) { var vParameters = vRequest.getParameters(true); var vParametersList = []; for (var vId in vParameters) { var paramValue = vParameters[vId]; if (paramValue instanceof Array) { for (var i=0; i<paramValue.length; i++) { vParametersList.push(encodeURIComponent(vId) + "=" + encodeURIComponent(paramValue[i])); } } else { vParametersList.push(encodeURIComponent(vId) + "=" + encodeURIComponent(paramValue)); } } if (vParametersList.length > 0) { value.setData(vParametersList.join("&")); } } else { value.setData(data); } value.setResponseType(vRequest.getResponseType()); value.addListener("sending", this._onsending, this); value.addListener("receiving", this._onreceiving, this); value.addListener("completed", this._oncompleted, this); value.addListener("aborted", this._onabort, this); value.addListener("timeout", this._ontimeout, this); value.addListener("failed", this._onfailed, this); } }, /** * Apply method for the state property. * * @param value {var} Current value * @param old {var} Previous value */ _applyState : function(value, old) { if (qx.core.Environment.get("qx.debug")) { if (qx.core.Environment.get("qx.debug.io.remote")) { this.debug("State: " + old + " => " + value); } } switch(value) { case "sending": this.fireEvent("sending"); break; case "receiving": this.fireEvent("receiving"); break; case "completed": case "aborted": case "timeout": case "failed": var vImpl = this.getImplementation(); if (!vImpl) { // implementation has already been disposed break; } // Disable future timeouts in case user handler blocks this.__disableRequestTimeout(); if (this.hasListener(value)) { var vResponse = qx.event.Registration.createEvent(value, qx.io.remote.Response); if (value == "completed") { var vContent = vImpl.getResponseContent(); vResponse.setContent(vContent); /* * Was there acceptable content? This might occur, for example, if * the web server was shut down unexpectedly and thus the connection * closed with no data having been sent. */ if (vContent === null) { // Nope. Change COMPLETED to FAILED. if (qx.core.Environment.get("qx.debug")) { if (qx.core.Environment.get("qx.debug.io.remote")) { this.debug("Altered State: " + value + " => failed"); } } value = "failed"; } } else if (value == "failed") { vResponse.setContent(vImpl.getResponseContent()); } vResponse.setStatusCode(vImpl.getStatusCode()); vResponse.setResponseHeaders(vImpl.getResponseHeaders()); this.dispatchEvent(vResponse); } // Disconnect and dispose implementation this.setImplementation(null); vImpl.dispose(); // Fire event to listeners //this.fireDataEvent(vEventType, vResponse); break; } } }, /* ***************************************************************************** ENVIRONMENT SETTINGS ***************************************************************************** */ environment : { "qx.debug.io.remote" : false, "qx.debug.io.remote.data" : false }, /* ***************************************************************************** DESTRUCTOR ***************************************************************************** */ destruct : function() { var vImpl = this.getImplementation(); if (vImpl) { this.setImplementation(null); vImpl.dispose(); } this.setRequest(null); } });