// Copyright 2006 The Closure Library Authors. All Rights Reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS-IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. /** * @fileoverview Definition of the ChannelRequest class. The request * object encapsulates the logic for making a single request, either for the * forward channel, back channel, or test channel, to the server. It contains * the logic for the two types of transports we use: * XMLHTTP and Image request. It provides timeout detection. More transports * to be added in future, such as Fetch, WebSocket. * * @visibility {:internal} */ goog.provide('goog.labs.net.webChannel.ChannelRequest'); goog.require('goog.Timer'); goog.require('goog.async.Throttle'); goog.require('goog.events.EventHandler'); goog.require('goog.labs.net.webChannel.Channel'); goog.require('goog.labs.net.webChannel.WebChannelDebug'); goog.require('goog.labs.net.webChannel.requestStats'); goog.require('goog.labs.net.webChannel.requestStats.ServerReachability'); goog.require('goog.labs.net.webChannel.requestStats.Stat'); goog.require('goog.net.ErrorCode'); goog.require('goog.net.EventType'); goog.require('goog.net.XmlHttp'); goog.require('goog.object'); goog.require('goog.userAgent'); /** * A new ChannelRequest is created for each request to the server. * * @param {goog.labs.net.webChannel.Channel} channel * The channel that owns this request. * @param {goog.labs.net.webChannel.WebChannelDebug} channelDebug A * WebChannelDebug to use for logging. * @param {string=} opt_sessionId The session id for the channel. * @param {string|number=} opt_requestId The request id for this request. * @param {number=} opt_retryId The retry id for this request. * @constructor * @struct * @final */ goog.labs.net.webChannel.ChannelRequest = function( channel, channelDebug, opt_sessionId, opt_requestId, opt_retryId) { /** * The channel object that owns the request. * @private {goog.labs.net.webChannel.Channel} */ this.channel_ = channel; /** * The channel debug to use for logging * @private {goog.labs.net.webChannel.WebChannelDebug} */ this.channelDebug_ = channelDebug; /** * The Session ID for the channel. * @private {string|undefined} */ this.sid_ = opt_sessionId; /** * The RID (request ID) for the request. * @private {string|number|undefined} */ this.rid_ = opt_requestId; /** * The attempt number of the current request. * @private {number} */ this.retryId_ = opt_retryId || 1; /** * An object to keep track of the channel request event listeners. * @private {!goog.events.EventHandler< * !goog.labs.net.webChannel.ChannelRequest>} */ this.eventHandler_ = new goog.events.EventHandler(this); /** * The timeout in ms before failing the request. * @private {number} */ this.timeout_ = goog.labs.net.webChannel.ChannelRequest.TIMEOUT_MS_; /** * A timer for polling responseText in browsers that don't fire * onreadystatechange during incremental loading of responseText. * @private {goog.Timer} */ this.pollingTimer_ = new goog.Timer(); this.pollingTimer_.setInterval( goog.labs.net.webChannel.ChannelRequest.POLLING_INTERVAL_MS_); /** * Extra HTTP headers to add to all the requests sent to the server. * @private {Object} */ this.extraHeaders_ = null; /** * Whether the request was successful. This is only set to true after the * request successfully completes. * @private {boolean} */ this.successful_ = false; /** * The TimerID of the timer used to detect if the request has timed-out. * @type {?number} * @private */ this.watchDogTimerId_ = null; /** * The time in the future when the request will timeout. * @private {?number} */ this.watchDogTimeoutTime_ = null; /** * The time the request started. * @private {?number} */ this.requestStartTime_ = null; /** * The type of request (XMLHTTP, IMG) * @private {?number} */ this.type_ = null; /** * The base Uri for the request. The includes all the parameters except the * one that indicates the retry number. * @private {goog.Uri} */ this.baseUri_ = null; /** * The request Uri that was actually used for the most recent request attempt. * @private {goog.Uri} */ this.requestUri_ = null; /** * The post data, if the request is a post. * @private {?string} */ this.postData_ = null; /** * The XhrLte request if the request is using XMLHTTP * @private {goog.net.XhrIo} */ this.xmlHttp_ = null; /** * The position of where the next unprocessed chunk starts in the response * text. * @private {number} */ this.xmlHttpChunkStart_ = 0; /** * The verb (Get or Post) for the request. * @private {?string} */ this.verb_ = null; /** * The last error if the request failed. * @private {?goog.labs.net.webChannel.ChannelRequest.Error} */ this.lastError_ = null; /** * The last status code received. * @private {number} */ this.lastStatusCode_ = -1; /** * Whether to send the Connection:close header as part of the request. * @private {boolean} */ this.sendClose_ = true; /** * Whether the request has been cancelled due to a call to cancel. * @private {boolean} */ this.cancelled_ = false; /** * A throttle time in ms for readystatechange events for the backchannel. * Useful for throttling when ready state is INTERACTIVE (partial data). * If set to zero no throttle is used. * * See WebChannelBase.prototype.readyStateChangeThrottleMs_ * * @private {number} */ this.readyStateChangeThrottleMs_ = 0; /** * The throttle for readystatechange events for the current request, or null * if there is none. * @private {goog.async.Throttle} */ this.readyStateChangeThrottle_ = null; /** * Whether to the result is expected to be encoded for chunking and thus * requires decoding. * @private {boolean} */ this.decodeChunks_ = false; }; goog.scope(function() { var Channel = goog.labs.net.webChannel.Channel; var ChannelRequest = goog.labs.net.webChannel.ChannelRequest; var requestStats = goog.labs.net.webChannel.requestStats; var WebChannelDebug = goog.labs.net.webChannel.WebChannelDebug; /** * Default timeout in MS for a request. The server must return data within this * time limit for the request to not timeout. * @private {number} */ ChannelRequest.TIMEOUT_MS_ = 45 * 1000; /** * How often to poll (in MS) for changes to responseText in browsers that don't * fire onreadystatechange during incremental loading of responseText. * @private {number} */ ChannelRequest.POLLING_INTERVAL_MS_ = 250; /** * Enum for channel requests type * @enum {number} * @private */ ChannelRequest.Type_ = { /** * XMLHTTP requests. */ XML_HTTP: 1, /** * IMG requests. */ CLOSE_REQUEST: 2 }; /** * Enum type for identifying an error. * @enum {number} */ ChannelRequest.Error = { /** * Errors due to a non-200 status code. */ STATUS: 0, /** * Errors due to no data being returned. */ NO_DATA: 1, /** * Errors due to a timeout. */ TIMEOUT: 2, /** * Errors due to the server returning an unknown. */ UNKNOWN_SESSION_ID: 3, /** * Errors due to bad data being received. */ BAD_DATA: 4, /** * Errors due to the handler throwing an exception. */ HANDLER_EXCEPTION: 5, /** * The browser declared itself offline during the request. */ BROWSER_OFFLINE: 6 }; /** * Returns a useful error string for debugging based on the specified error * code. * @param {?ChannelRequest.Error} errorCode The error code. * @param {number} statusCode The HTTP status code. * @return {string} The error string for the given code combination. */ ChannelRequest.errorStringFromCode = function(errorCode, statusCode) { switch (errorCode) { case ChannelRequest.Error.STATUS: return 'Non-200 return code (' + statusCode + ')'; case ChannelRequest.Error.NO_DATA: return 'XMLHTTP failure (no data)'; case ChannelRequest.Error.TIMEOUT: return 'HttpConnection timeout'; default: return 'Unknown error'; } }; /** * Sentinel value used to indicate an invalid chunk in a multi-chunk response. * @private {Object} */ ChannelRequest.INVALID_CHUNK_ = {}; /** * Sentinel value used to indicate an incomplete chunk in a multi-chunk * response. * @private {Object} */ ChannelRequest.INCOMPLETE_CHUNK_ = {}; /** * Returns whether XHR streaming is supported on this browser. * * @return {boolean} Whether XHR streaming is supported. * @see http://code.google.com/p/closure-library/issues/detail?id=346 */ ChannelRequest.supportsXhrStreaming = function() { return !goog.userAgent.IE || goog.userAgent.isDocumentModeOrHigher(10); }; /** * Sets extra HTTP headers to add to all the requests sent to the server. * * @param {Object} extraHeaders The HTTP headers. */ ChannelRequest.prototype.setExtraHeaders = function(extraHeaders) { this.extraHeaders_ = extraHeaders; }; /** * Sets the timeout for a request * * @param {number} timeout The timeout in MS for when we fail the request. */ ChannelRequest.prototype.setTimeout = function(timeout) { this.timeout_ = timeout; }; /** * Sets the throttle for handling onreadystatechange events for the request. * * @param {number} throttle The throttle in ms. A value of zero indicates * no throttle. */ ChannelRequest.prototype.setReadyStateChangeThrottle = function(throttle) { this.readyStateChangeThrottleMs_ = throttle; }; /** * Uses XMLHTTP to send an HTTP POST to the server. * * @param {goog.Uri} uri The uri of the request. * @param {string} postData The data for the post body. * @param {boolean} decodeChunks Whether to the result is expected to be * encoded for chunking and thus requires decoding. */ ChannelRequest.prototype.xmlHttpPost = function(uri, postData, decodeChunks) { this.type_ = ChannelRequest.Type_.XML_HTTP; this.baseUri_ = uri.clone().makeUnique(); this.postData_ = postData; this.decodeChunks_ = decodeChunks; this.sendXmlHttp_(null /* hostPrefix */); }; /** * Uses XMLHTTP to send an HTTP GET to the server. * * @param {goog.Uri} uri The uri of the request. * @param {boolean} decodeChunks Whether to the result is expected to be * encoded for chunking and thus requires decoding. * @param {?string} hostPrefix The host prefix, if we might be using a * secondary domain. Note that it should also be in the URL, adding this * won't cause it to be added to the URL. * @param {boolean=} opt_noClose Whether to request that the tcp/ip connection * should be closed. */ ChannelRequest.prototype.xmlHttpGet = function( uri, decodeChunks, hostPrefix, opt_noClose) { this.type_ = ChannelRequest.Type_.XML_HTTP; this.baseUri_ = uri.clone().makeUnique(); this.postData_ = null; this.decodeChunks_ = decodeChunks; if (opt_noClose) { this.sendClose_ = false; } this.sendXmlHttp_(hostPrefix); }; /** * Sends a request via XMLHTTP according to the current state of the request * object. * * @param {?string} hostPrefix The host prefix, if we might be using a secondary * domain. * @private */ ChannelRequest.prototype.sendXmlHttp_ = function(hostPrefix) { this.requestStartTime_ = goog.now(); this.ensureWatchDogTimer_(); // clone the base URI to create the request URI. The request uri has the // attempt number as a parameter which helps in debugging. this.requestUri_ = this.baseUri_.clone(); this.requestUri_.setParameterValues('t', this.retryId_); // send the request either as a POST or GET this.xmlHttpChunkStart_ = 0; var useSecondaryDomains = this.channel_.shouldUseSecondaryDomains(); this.xmlHttp_ = this.channel_.createXhrIo(useSecondaryDomains ? hostPrefix : null); if (this.readyStateChangeThrottleMs_ > 0) { this.readyStateChangeThrottle_ = new goog.async.Throttle( goog.bind(this.xmlHttpHandler_, this, this.xmlHttp_), this.readyStateChangeThrottleMs_); } this.eventHandler_.listen( this.xmlHttp_, goog.net.EventType.READY_STATE_CHANGE, this.readyStateChangeHandler_); var headers = this.extraHeaders_ ? goog.object.clone(this.extraHeaders_) : {}; if (this.postData_) { this.verb_ = 'POST'; headers['Content-Type'] = 'application/x-www-form-urlencoded'; this.xmlHttp_.send(this.requestUri_, this.verb_, this.postData_, headers); } else { this.verb_ = 'GET'; // If the user agent is webkit, we cannot send the close header since it is // disallowed by the browser. If we attempt to set the "Connection: close" // header in WEBKIT browser, it will actually causes an error message. if (this.sendClose_ && !goog.userAgent.WEBKIT) { headers['Connection'] = 'close'; } this.xmlHttp_.send(this.requestUri_, this.verb_, null, headers); } requestStats.notifyServerReachabilityEvent( requestStats.ServerReachability.REQUEST_MADE); this.channelDebug_.xmlHttpChannelRequest( this.verb_, this.requestUri_, this.rid_, this.retryId_, this.postData_); }; /** * Handles a readystatechange event. * @param {goog.events.Event} evt The event. * @private */ ChannelRequest.prototype.readyStateChangeHandler_ = function(evt) { var xhr = /** @type {goog.net.XhrIo} */ (evt.target); var throttle = this.readyStateChangeThrottle_; if (throttle && xhr.getReadyState() == goog.net.XmlHttp.ReadyState.INTERACTIVE) { // Only throttle in the partial data case. this.channelDebug_.debug('Throttling readystatechange.'); throttle.fire(); } else { // If we haven't throttled, just handle response directly. this.xmlHttpHandler_(xhr); } }; /** * XmlHttp handler * @param {goog.net.XhrIo} xmlhttp The XhrIo object for the current request. * @private */ ChannelRequest.prototype.xmlHttpHandler_ = function(xmlhttp) { requestStats.onStartExecution(); try { if (xmlhttp == this.xmlHttp_) { this.onXmlHttpReadyStateChanged_(); } else { this.channelDebug_.warning( 'Called back with an ' + 'unexpected xmlhttp'); } } catch (ex) { this.channelDebug_.debug('Failed call to OnXmlHttpReadyStateChanged_'); if (this.xmlHttp_ && this.xmlHttp_.getResponseText()) { this.channelDebug_.dumpException( ex, 'ResponseText: ' + this.xmlHttp_.getResponseText()); } else { this.channelDebug_.dumpException(ex, 'No response text'); } } finally { requestStats.onEndExecution(); } }; /** * Called by the readystate handler for XMLHTTP requests. * * @private */ ChannelRequest.prototype.onXmlHttpReadyStateChanged_ = function() { var readyState = this.xmlHttp_.getReadyState(); var errorCode = this.xmlHttp_.getLastErrorCode(); var statusCode = this.xmlHttp_.getStatus(); // we get partial results in browsers that support ready state interactive. // We also make sure that getResponseText is not null in interactive mode // before we continue. However, we don't do it in Opera because it only // fire readyState == INTERACTIVE once. We need the following code to poll if (readyState < goog.net.XmlHttp.ReadyState.INTERACTIVE || readyState == goog.net.XmlHttp.ReadyState.INTERACTIVE && !goog.userAgent.OPERA && !this.xmlHttp_.getResponseText()) { // not yet ready return; } // Dispatch any appropriate network events. if (!this.cancelled_ && readyState == goog.net.XmlHttp.ReadyState.COMPLETE && errorCode != goog.net.ErrorCode.ABORT) { // Pretty conservative, these are the only known scenarios which we'd // consider indicative of a truly non-functional network connection. if (errorCode == goog.net.ErrorCode.TIMEOUT || statusCode <= 0) { requestStats.notifyServerReachabilityEvent( requestStats.ServerReachability.REQUEST_FAILED); } else { requestStats.notifyServerReachabilityEvent( requestStats.ServerReachability.REQUEST_SUCCEEDED); } } // got some data so cancel the watchdog timer this.cancelWatchDogTimer_(); var status = this.xmlHttp_.getStatus(); this.lastStatusCode_ = status; var responseText = this.xmlHttp_.getResponseText(); if (!responseText) { this.channelDebug_.debug( 'No response text for uri ' + this.requestUri_ + ' status ' + status); } this.successful_ = (status == 200); this.channelDebug_.xmlHttpChannelResponseMetaData( /** @type {string} */ (this.verb_), this.requestUri_, this.rid_, this.retryId_, readyState, status); if (!this.successful_) { if (status == 400 && responseText.indexOf('Unknown SID') > 0) { // the server error string will include 'Unknown SID' which indicates the // server doesn't know about the session (maybe it got restarted, maybe // the user got moved to another server, etc.,). Handlers can special // case this error this.lastError_ = ChannelRequest.Error.UNKNOWN_SESSION_ID; requestStats.notifyStatEvent( requestStats.Stat.REQUEST_UNKNOWN_SESSION_ID); this.channelDebug_.warning('XMLHTTP Unknown SID (' + this.rid_ + ')'); } else { this.lastError_ = ChannelRequest.Error.STATUS; requestStats.notifyStatEvent(requestStats.Stat.REQUEST_BAD_STATUS); this.channelDebug_.warning( 'XMLHTTP Bad status ' + status + ' (' + this.rid_ + ')'); } this.cleanup_(); this.dispatchFailure_(); return; } if (this.decodeChunks_) { this.decodeNextChunks_(readyState, responseText); if (goog.userAgent.OPERA && this.successful_ && readyState == goog.net.XmlHttp.ReadyState.INTERACTIVE) { this.startPolling_(); } } else { this.channelDebug_.xmlHttpChannelResponseText( this.rid_, responseText, null); this.safeOnRequestData_(responseText); } if (readyState == goog.net.XmlHttp.ReadyState.COMPLETE) { this.cleanup_(); } if (!this.successful_) { return; } if (!this.cancelled_) { if (readyState == goog.net.XmlHttp.ReadyState.COMPLETE) { this.channel_.onRequestComplete(this); } else { // The default is false, the result from this callback shouldn't carry // over to the next callback, otherwise the request looks successful if // the watchdog timer gets called this.successful_ = false; this.ensureWatchDogTimer_(); } } }; /** * Decodes the next set of available chunks in the response. * @param {number} readyState The value of readyState. * @param {string} responseText The value of responseText. * @private */ ChannelRequest.prototype.decodeNextChunks_ = function( readyState, responseText) { var decodeNextChunksSuccessful = true; while (!this.cancelled_ && this.xmlHttpChunkStart_ < responseText.length) { var chunkText = this.getNextChunk_(responseText); if (chunkText == ChannelRequest.INCOMPLETE_CHUNK_) { if (readyState == goog.net.XmlHttp.ReadyState.COMPLETE) { // should have consumed entire response when the request is done this.lastError_ = ChannelRequest.Error.BAD_DATA; requestStats.notifyStatEvent(requestStats.Stat.REQUEST_INCOMPLETE_DATA); decodeNextChunksSuccessful = false; } this.channelDebug_.xmlHttpChannelResponseText( this.rid_, null, '[Incomplete Response]'); break; } else if (chunkText == ChannelRequest.INVALID_CHUNK_) { this.lastError_ = ChannelRequest.Error.BAD_DATA; requestStats.notifyStatEvent(requestStats.Stat.REQUEST_BAD_DATA); this.channelDebug_.xmlHttpChannelResponseText( this.rid_, responseText, '[Invalid Chunk]'); decodeNextChunksSuccessful = false; break; } else { this.channelDebug_.xmlHttpChannelResponseText( this.rid_, /** @type {string} */ (chunkText), null); this.safeOnRequestData_(/** @type {string} */ (chunkText)); } } if (readyState == goog.net.XmlHttp.ReadyState.COMPLETE && responseText.length == 0) { // also an error if we didn't get any response this.lastError_ = ChannelRequest.Error.NO_DATA; requestStats.notifyStatEvent(requestStats.Stat.REQUEST_NO_DATA); decodeNextChunksSuccessful = false; } this.successful_ = this.successful_ && decodeNextChunksSuccessful; if (!decodeNextChunksSuccessful) { // malformed response - we make this trigger retry logic this.channelDebug_.xmlHttpChannelResponseText( this.rid_, responseText, '[Invalid Chunked Response]'); this.cleanup_(); this.dispatchFailure_(); } }; /** * Polls the response for new data. * @private */ ChannelRequest.prototype.pollResponse_ = function() { var readyState = this.xmlHttp_.getReadyState(); var responseText = this.xmlHttp_.getResponseText(); if (this.xmlHttpChunkStart_ < responseText.length) { this.cancelWatchDogTimer_(); this.decodeNextChunks_(readyState, responseText); if (this.successful_ && readyState != goog.net.XmlHttp.ReadyState.COMPLETE) { this.ensureWatchDogTimer_(); } } }; /** * Starts a polling interval for changes to responseText of the * XMLHttpRequest, for browsers that don't fire onreadystatechange * as data comes in incrementally. This timer is disabled in * cleanup_(). * @private */ ChannelRequest.prototype.startPolling_ = function() { this.eventHandler_.listen( this.pollingTimer_, goog.Timer.TICK, this.pollResponse_); this.pollingTimer_.start(); }; /** * Returns the next chunk of a chunk-encoded response. This is not standard * HTTP chunked encoding because browsers don't expose the chunk boundaries to * the application through XMLHTTP. So we have an additional chunk encoding at * the application level that lets us tell where the beginning and end of * individual responses are so that we can only try to eval a complete JS array. * * The encoding is the size of the chunk encoded as a decimal string followed * by a newline followed by the data. * * @param {string} responseText The response text from the XMLHTTP response. * @return {string|Object} The next chunk string or a sentinel object * indicating a special condition. * @private */ ChannelRequest.prototype.getNextChunk_ = function(responseText) { var sizeStartIndex = this.xmlHttpChunkStart_; var sizeEndIndex = responseText.indexOf('\n', sizeStartIndex); if (sizeEndIndex == -1) { return ChannelRequest.INCOMPLETE_CHUNK_; } var sizeAsString = responseText.substring(sizeStartIndex, sizeEndIndex); var size = Number(sizeAsString); if (isNaN(size)) { return ChannelRequest.INVALID_CHUNK_; } var chunkStartIndex = sizeEndIndex + 1; if (chunkStartIndex + size > responseText.length) { return ChannelRequest.INCOMPLETE_CHUNK_; } var chunkText = responseText.substr(chunkStartIndex, size); this.xmlHttpChunkStart_ = chunkStartIndex + size; return chunkText; }; /** * Uses an IMG tag or navigator.sendBeacon to send an HTTP get to the server. * * This is only currently used to terminate the connection, as an IMG tag is * the most reliable way to send something to the server while the page * is getting torn down. * * Navigator.sendBeacon is available on Chrome and Firefox as a formal * solution to ensure delivery without blocking window close. See * https://developer.mozilla.org/en-US/docs/Web/API/Navigator/sendBeacon * * For Chrome Apps, sendBeacon is always necessary due to Content Security * Policy (CSP) violation of using an IMG tag. * * @param {goog.Uri} uri The uri to send a request to. */ ChannelRequest.prototype.sendCloseRequest = function(uri) { this.type_ = ChannelRequest.Type_.CLOSE_REQUEST; this.baseUri_ = uri.clone().makeUnique(); var requestSent = false; if (goog.global.navigator && goog.global.navigator.sendBeacon) { // empty string body to avoid 413 error on chrome < 41 requestSent = goog.global.navigator.sendBeacon(this.baseUri_.toString(), ''); } if (!requestSent) { var eltImg = new Image(); eltImg.src = this.baseUri_; } this.requestStartTime_ = goog.now(); this.ensureWatchDogTimer_(); }; /** * Cancels the request no matter what the underlying transport is. */ ChannelRequest.prototype.cancel = function() { this.cancelled_ = true; this.cleanup_(); }; /** * Ensures that there is watchdog timeout which is used to ensure that * the connection completes in time. * * @private */ ChannelRequest.prototype.ensureWatchDogTimer_ = function() { this.watchDogTimeoutTime_ = goog.now() + this.timeout_; this.startWatchDogTimer_(this.timeout_); }; /** * Starts the watchdog timer which is used to ensure that the connection * completes in time. * @param {number} time The number of milliseconds to wait. * @private */ ChannelRequest.prototype.startWatchDogTimer_ = function(time) { if (this.watchDogTimerId_ != null) { // assertion throw Error('WatchDog timer not null'); } this.watchDogTimerId_ = requestStats.setTimeout(goog.bind(this.onWatchDogTimeout_, this), time); }; /** * Cancels the watchdog timer if it has been started. * * @private */ ChannelRequest.prototype.cancelWatchDogTimer_ = function() { if (this.watchDogTimerId_) { goog.global.clearTimeout(this.watchDogTimerId_); this.watchDogTimerId_ = null; } }; /** * Called when the watchdog timer is triggered. It also handles a case where it * is called too early which we suspect may be happening sometimes * (not sure why) * * @private */ ChannelRequest.prototype.onWatchDogTimeout_ = function() { this.watchDogTimerId_ = null; var now = goog.now(); if (now - this.watchDogTimeoutTime_ >= 0) { this.handleTimeout_(); } else { // got called too early for some reason this.channelDebug_.warning('WatchDog timer called too early'); this.startWatchDogTimer_(this.watchDogTimeoutTime_ - now); } }; /** * Called when the request has actually timed out. Will cleanup and notify the * channel of the failure. * * @private */ ChannelRequest.prototype.handleTimeout_ = function() { if (this.successful_) { // Should never happen. this.channelDebug_.severe( 'Received watchdog timeout even though request loaded successfully'); } this.channelDebug_.timeoutResponse(this.requestUri_); // IMG or SendBeacon requests never notice if they were successful, // and always 'time out'. This fact says nothing about reachability. if (this.type_ != ChannelRequest.Type_.CLOSE_REQUEST) { requestStats.notifyServerReachabilityEvent( requestStats.ServerReachability.REQUEST_FAILED); requestStats.notifyStatEvent(requestStats.Stat.REQUEST_TIMEOUT); } this.cleanup_(); // Set error and dispatch failure. // This is called for CLOSE_REQUEST too to ensure channel_.onRequestComplete. this.lastError_ = ChannelRequest.Error.TIMEOUT; this.dispatchFailure_(); }; /** * Notifies the channel that this request failed. * @private */ ChannelRequest.prototype.dispatchFailure_ = function() { if (this.channel_.isClosed() || this.cancelled_) { return; } this.channel_.onRequestComplete(this); }; /** * Cleans up the objects used to make the request. This function is * idempotent. * * @private */ ChannelRequest.prototype.cleanup_ = function() { this.cancelWatchDogTimer_(); goog.dispose(this.readyStateChangeThrottle_); this.readyStateChangeThrottle_ = null; // Stop the polling timer, if necessary. this.pollingTimer_.stop(); // Unhook all event handlers. this.eventHandler_.removeAll(); if (this.xmlHttp_) { // clear out this.xmlHttp_ before aborting so we handle getting reentered // inside abort var xmlhttp = this.xmlHttp_; this.xmlHttp_ = null; xmlhttp.abort(); xmlhttp.dispose(); } }; /** * Indicates whether the request was successful. Only valid after the handler * is called to indicate completion of the request. * * @return {boolean} True if the request succeeded. */ ChannelRequest.prototype.getSuccess = function() { return this.successful_; }; /** * If the request was not successful, returns the reason. * * @return {?ChannelRequest.Error} The last error. */ ChannelRequest.prototype.getLastError = function() { return this.lastError_; }; /** * Returns the status code of the last request. * @return {number} The status code of the last request. */ ChannelRequest.prototype.getLastStatusCode = function() { return this.lastStatusCode_; }; /** * Returns the session id for this channel. * * @return {string|undefined} The session ID. */ ChannelRequest.prototype.getSessionId = function() { return this.sid_; }; /** * Returns the request id for this request. Each request has a unique request * id and the request IDs are a sequential increasing count. * * @return {string|number|undefined} The request ID. */ ChannelRequest.prototype.getRequestId = function() { return this.rid_; }; /** * Returns the data for a post, if this request is a post. * * @return {?string} The POST data provided by the request initiator. */ ChannelRequest.prototype.getPostData = function() { return this.postData_; }; /** * Returns the XhrIo request object. * * @return {?goog.net.XhrIo} Any XhrIo request created for this object. */ ChannelRequest.prototype.getXhr = function() { return this.xmlHttp_; }; /** * Returns the time that the request started, if it has started. * * @return {?number} The time the request started, as returned by goog.now(). */ ChannelRequest.prototype.getRequestStartTime = function() { return this.requestStartTime_; }; /** * Helper to call the callback's onRequestData, which catches any * exception and cleans up the request. * @param {string} data The request data. * @private */ ChannelRequest.prototype.safeOnRequestData_ = function(data) { try { this.channel_.onRequestData(this, data); var stats = requestStats.ServerReachability; requestStats.notifyServerReachabilityEvent(stats.BACK_CHANNEL_ACTIVITY); } catch (e) { // Dump debug info, but keep going without closing the channel. this.channelDebug_.dumpException(e, 'Error in httprequest callback'); } }; /** * Convenience factory method. * * @param {Channel} channel The channel object that owns this request. * @param {WebChannelDebug} channelDebug A WebChannelDebug to use for logging. * @param {string=} opt_sessionId The session id for the channel. * @param {string|number=} opt_requestId The request id for this request. * @param {number=} opt_retryId The retry id for this request. * @return {!ChannelRequest} The created channel request. */ ChannelRequest.createChannelRequest = function( channel, channelDebug, opt_sessionId, opt_requestId, opt_retryId) { return new ChannelRequest( channel, channelDebug, opt_sessionId, opt_requestId, opt_retryId); }; }); // goog.scope