You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							1364 lines
						
					
					
						
							38 KiB
						
					
					
				
			
		
		
	
	
							1364 lines
						
					
					
						
							38 KiB
						
					
					
				/** | 
						|
 * HTTP client-side implementation that uses forge.net sockets. | 
						|
 * | 
						|
 * @author Dave Longley | 
						|
 * | 
						|
 * Copyright (c) 2010-2014 Digital Bazaar, Inc. All rights reserved. | 
						|
 */ | 
						|
var forge = require('./forge'); | 
						|
require('./debug'); | 
						|
require('./tls'); | 
						|
require('./util'); | 
						|
 | 
						|
// define http namespace | 
						|
var http = module.exports = forge.http = forge.http || {}; | 
						|
 | 
						|
// logging category | 
						|
var cat = 'forge.http'; | 
						|
 | 
						|
// add array of clients to debug storage | 
						|
if(forge.debug) { | 
						|
  forge.debug.set('forge.http', 'clients', []); | 
						|
} | 
						|
 | 
						|
// normalizes an http header field name | 
						|
var _normalize = function(name) { | 
						|
  return name.toLowerCase().replace(/(^.)|(-.)/g, | 
						|
    function(a) {return a.toUpperCase();}); | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Gets the local storage ID for the given client. | 
						|
 * | 
						|
 * @param client the client to get the local storage ID for. | 
						|
 * | 
						|
 * @return the local storage ID to use. | 
						|
 */ | 
						|
var _getStorageId = function(client) { | 
						|
  // TODO: include browser in ID to avoid sharing cookies between | 
						|
  // browsers (if this is undesirable) | 
						|
  // navigator.userAgent | 
						|
  return 'forge.http.' + | 
						|
    client.url.scheme + '.' + | 
						|
    client.url.host + '.' + | 
						|
    client.url.port; | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Loads persistent cookies from disk for the given client. | 
						|
 * | 
						|
 * @param client the client. | 
						|
 */ | 
						|
var _loadCookies = function(client) { | 
						|
  if(client.persistCookies) { | 
						|
    try { | 
						|
      var cookies = forge.util.getItem( | 
						|
        client.socketPool.flashApi, | 
						|
        _getStorageId(client), 'cookies'); | 
						|
      client.cookies = cookies || {}; | 
						|
    } catch(ex) { | 
						|
      // no flash storage available, just silently fail | 
						|
      // TODO: i assume we want this logged somewhere or | 
						|
      // should it actually generate an error | 
						|
      //forge.log.error(cat, ex); | 
						|
    } | 
						|
  } | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Saves persistent cookies on disk for the given client. | 
						|
 * | 
						|
 * @param client the client. | 
						|
 */ | 
						|
var _saveCookies = function(client) { | 
						|
  if(client.persistCookies) { | 
						|
    try { | 
						|
      forge.util.setItem( | 
						|
        client.socketPool.flashApi, | 
						|
        _getStorageId(client), 'cookies', client.cookies); | 
						|
    } catch(ex) { | 
						|
      // no flash storage available, just silently fail | 
						|
      // TODO: i assume we want this logged somewhere or | 
						|
      // should it actually generate an error | 
						|
      //forge.log.error(cat, ex); | 
						|
    } | 
						|
  } | 
						|
 | 
						|
  // FIXME: remove me | 
						|
  _loadCookies(client); | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Clears persistent cookies on disk for the given client. | 
						|
 * | 
						|
 * @param client the client. | 
						|
 */ | 
						|
var _clearCookies = function(client) { | 
						|
  if(client.persistCookies) { | 
						|
    try { | 
						|
      // only thing stored is 'cookies', so clear whole storage | 
						|
      forge.util.clearItems( | 
						|
        client.socketPool.flashApi, | 
						|
        _getStorageId(client)); | 
						|
    } catch(ex) { | 
						|
      // no flash storage available, just silently fail | 
						|
      // TODO: i assume we want this logged somewhere or | 
						|
      // should it actually generate an error | 
						|
      //forge.log.error(cat, ex); | 
						|
    } | 
						|
  } | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Connects and sends a request. | 
						|
 * | 
						|
 * @param client the http client. | 
						|
 * @param socket the socket to use. | 
						|
 */ | 
						|
var _doRequest = function(client, socket) { | 
						|
  if(socket.isConnected()) { | 
						|
    // already connected | 
						|
    socket.options.request.connectTime = +new Date(); | 
						|
    socket.connected({ | 
						|
      type: 'connect', | 
						|
      id: socket.id | 
						|
    }); | 
						|
  } else { | 
						|
    // connect | 
						|
    socket.options.request.connectTime = +new Date(); | 
						|
    socket.connect({ | 
						|
      host: client.url.host, | 
						|
      port: client.url.port, | 
						|
      policyPort: client.policyPort, | 
						|
      policyUrl: client.policyUrl | 
						|
    }); | 
						|
  } | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Handles the next request or marks a socket as idle. | 
						|
 * | 
						|
 * @param client the http client. | 
						|
 * @param socket the socket. | 
						|
 */ | 
						|
var _handleNextRequest = function(client, socket) { | 
						|
  // clear buffer | 
						|
  socket.buffer.clear(); | 
						|
 | 
						|
  // get pending request | 
						|
  var pending = null; | 
						|
  while(pending === null && client.requests.length > 0) { | 
						|
    pending = client.requests.shift(); | 
						|
    if(pending.request.aborted) { | 
						|
      pending = null; | 
						|
    } | 
						|
  } | 
						|
 | 
						|
  // mark socket idle if no pending requests | 
						|
  if(pending === null) { | 
						|
    if(socket.options !== null) { | 
						|
      socket.options = null; | 
						|
    } | 
						|
    client.idle.push(socket); | 
						|
  } else { | 
						|
    // handle pending request, allow 1 retry | 
						|
    socket.retries = 1; | 
						|
    socket.options = pending; | 
						|
    _doRequest(client, socket); | 
						|
  } | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Sets up a socket for use with an http client. | 
						|
 * | 
						|
 * @param client the parent http client. | 
						|
 * @param socket the socket to set up. | 
						|
 * @param tlsOptions if the socket must use TLS, the TLS options. | 
						|
 */ | 
						|
var _initSocket = function(client, socket, tlsOptions) { | 
						|
  // no socket options yet | 
						|
  socket.options = null; | 
						|
 | 
						|
  // set up handlers | 
						|
  socket.connected = function(e) { | 
						|
    // socket primed by caching TLS session, handle next request | 
						|
    if(socket.options === null) { | 
						|
      _handleNextRequest(client, socket); | 
						|
    } else { | 
						|
      // socket in use | 
						|
      var request = socket.options.request; | 
						|
      request.connectTime = +new Date() - request.connectTime; | 
						|
      e.socket = socket; | 
						|
      socket.options.connected(e); | 
						|
      if(request.aborted) { | 
						|
        socket.close(); | 
						|
      } else { | 
						|
        var out = request.toString(); | 
						|
        if(request.body) { | 
						|
          out += request.body; | 
						|
        } | 
						|
        request.time = +new Date(); | 
						|
        socket.send(out); | 
						|
        request.time = +new Date() - request.time; | 
						|
        socket.options.response.time = +new Date(); | 
						|
        socket.sending = true; | 
						|
      } | 
						|
    } | 
						|
  }; | 
						|
  socket.closed = function(e) { | 
						|
    if(socket.sending) { | 
						|
      socket.sending = false; | 
						|
      if(socket.retries > 0) { | 
						|
        --socket.retries; | 
						|
        _doRequest(client, socket); | 
						|
      } else { | 
						|
        // error, closed during send | 
						|
        socket.error({ | 
						|
          id: socket.id, | 
						|
          type: 'ioError', | 
						|
          message: 'Connection closed during send. Broken pipe.', | 
						|
          bytesAvailable: 0 | 
						|
        }); | 
						|
      } | 
						|
    } else { | 
						|
      // handle unspecified content-length transfer | 
						|
      var response = socket.options.response; | 
						|
      if(response.readBodyUntilClose) { | 
						|
        response.time = +new Date() - response.time; | 
						|
        response.bodyReceived = true; | 
						|
        socket.options.bodyReady({ | 
						|
          request: socket.options.request, | 
						|
          response: response, | 
						|
          socket: socket | 
						|
        }); | 
						|
      } | 
						|
      socket.options.closed(e); | 
						|
      _handleNextRequest(client, socket); | 
						|
    } | 
						|
  }; | 
						|
  socket.data = function(e) { | 
						|
    socket.sending = false; | 
						|
    var request = socket.options.request; | 
						|
    if(request.aborted) { | 
						|
      socket.close(); | 
						|
    } else { | 
						|
      // receive all bytes available | 
						|
      var response = socket.options.response; | 
						|
      var bytes = socket.receive(e.bytesAvailable); | 
						|
      if(bytes !== null) { | 
						|
        // receive header and then body | 
						|
        socket.buffer.putBytes(bytes); | 
						|
        if(!response.headerReceived) { | 
						|
          response.readHeader(socket.buffer); | 
						|
          if(response.headerReceived) { | 
						|
            socket.options.headerReady({ | 
						|
              request: socket.options.request, | 
						|
              response: response, | 
						|
              socket: socket | 
						|
            }); | 
						|
          } | 
						|
        } | 
						|
        if(response.headerReceived && !response.bodyReceived) { | 
						|
          response.readBody(socket.buffer); | 
						|
        } | 
						|
        if(response.bodyReceived) { | 
						|
          socket.options.bodyReady({ | 
						|
            request: socket.options.request, | 
						|
            response: response, | 
						|
            socket: socket | 
						|
          }); | 
						|
          // close connection if requested or by default on http/1.0 | 
						|
          var value = response.getField('Connection') || ''; | 
						|
          if(value.indexOf('close') != -1 || | 
						|
            (response.version === 'HTTP/1.0' && | 
						|
            response.getField('Keep-Alive') === null)) { | 
						|
            socket.close(); | 
						|
          } else { | 
						|
            _handleNextRequest(client, socket); | 
						|
          } | 
						|
        } | 
						|
      } | 
						|
    } | 
						|
  }; | 
						|
  socket.error = function(e) { | 
						|
    // do error callback, include request | 
						|
    socket.options.error({ | 
						|
      type: e.type, | 
						|
      message: e.message, | 
						|
      request: socket.options.request, | 
						|
      response: socket.options.response, | 
						|
      socket: socket | 
						|
    }); | 
						|
    socket.close(); | 
						|
  }; | 
						|
 | 
						|
  // wrap socket for TLS | 
						|
  if(tlsOptions) { | 
						|
    socket = forge.tls.wrapSocket({ | 
						|
      sessionId: null, | 
						|
      sessionCache: {}, | 
						|
      caStore: tlsOptions.caStore, | 
						|
      cipherSuites: tlsOptions.cipherSuites, | 
						|
      socket: socket, | 
						|
      virtualHost: tlsOptions.virtualHost, | 
						|
      verify: tlsOptions.verify, | 
						|
      getCertificate: tlsOptions.getCertificate, | 
						|
      getPrivateKey: tlsOptions.getPrivateKey, | 
						|
      getSignature: tlsOptions.getSignature, | 
						|
      deflate: tlsOptions.deflate || null, | 
						|
      inflate: tlsOptions.inflate || null | 
						|
    }); | 
						|
 | 
						|
    socket.options = null; | 
						|
    socket.buffer = forge.util.createBuffer(); | 
						|
    client.sockets.push(socket); | 
						|
    if(tlsOptions.prime) { | 
						|
      // prime socket by connecting and caching TLS session, will do | 
						|
      // next request from there | 
						|
      socket.connect({ | 
						|
        host: client.url.host, | 
						|
        port: client.url.port, | 
						|
        policyPort: client.policyPort, | 
						|
        policyUrl: client.policyUrl | 
						|
      }); | 
						|
    } else { | 
						|
      // do not prime socket, just add as idle | 
						|
      client.idle.push(socket); | 
						|
    } | 
						|
  } else { | 
						|
    // no need to prime non-TLS sockets | 
						|
    socket.buffer = forge.util.createBuffer(); | 
						|
    client.sockets.push(socket); | 
						|
    client.idle.push(socket); | 
						|
  } | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Checks to see if the given cookie has expired. If the cookie's max-age | 
						|
 * plus its created time is less than the time now, it has expired, unless | 
						|
 * its max-age is set to -1 which indicates it will never expire. | 
						|
 * | 
						|
 * @param cookie the cookie to check. | 
						|
 * | 
						|
 * @return true if it has expired, false if not. | 
						|
 */ | 
						|
var _hasCookieExpired = function(cookie) { | 
						|
  var rval = false; | 
						|
 | 
						|
  if(cookie.maxAge !== -1) { | 
						|
    var now = _getUtcTime(new Date()); | 
						|
    var expires = cookie.created + cookie.maxAge; | 
						|
    if(expires <= now) { | 
						|
      rval = true; | 
						|
    } | 
						|
  } | 
						|
 | 
						|
  return rval; | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Adds cookies in the given client to the given request. | 
						|
 * | 
						|
 * @param client the client. | 
						|
 * @param request the request. | 
						|
 */ | 
						|
var _writeCookies = function(client, request) { | 
						|
  var expired = []; | 
						|
  var url = client.url; | 
						|
  var cookies = client.cookies; | 
						|
  for(var name in cookies) { | 
						|
    // get cookie paths | 
						|
    var paths = cookies[name]; | 
						|
    for(var p in paths) { | 
						|
      var cookie = paths[p]; | 
						|
      if(_hasCookieExpired(cookie)) { | 
						|
        // store for clean up | 
						|
        expired.push(cookie); | 
						|
      } else if(request.path.indexOf(cookie.path) === 0) { | 
						|
        // path or path's ancestor must match cookie.path | 
						|
        request.addCookie(cookie); | 
						|
      } | 
						|
    } | 
						|
  } | 
						|
 | 
						|
  // clean up expired cookies | 
						|
  for(var i = 0; i < expired.length; ++i) { | 
						|
    var cookie = expired[i]; | 
						|
    client.removeCookie(cookie.name, cookie.path); | 
						|
  } | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Gets cookies from the given response and adds the to the given client. | 
						|
 * | 
						|
 * @param client the client. | 
						|
 * @param response the response. | 
						|
 */ | 
						|
var _readCookies = function(client, response) { | 
						|
  var cookies = response.getCookies(); | 
						|
  for(var i = 0; i < cookies.length; ++i) { | 
						|
    try { | 
						|
      client.setCookie(cookies[i]); | 
						|
    } catch(ex) { | 
						|
      // ignore failure to add other-domain, etc. cookies | 
						|
    } | 
						|
  } | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Creates an http client that uses forge.net sockets as a backend and | 
						|
 * forge.tls for security. | 
						|
 * | 
						|
 * @param options: | 
						|
 *   url: the url to connect to (scheme://host:port). | 
						|
 *     socketPool: the flash socket pool to use. | 
						|
 *   policyPort: the flash policy port to use (if other than the | 
						|
 *     socket pool default), use 0 for flash default. | 
						|
 *   policyUrl: the flash policy file URL to use (if provided will | 
						|
 *     be used instead of a policy port). | 
						|
 *   connections: number of connections to use to handle requests. | 
						|
 *   caCerts: an array of certificates to trust for TLS, certs may | 
						|
 *     be PEM-formatted or cert objects produced via forge.pki. | 
						|
 *   cipherSuites: an optional array of cipher suites to use, | 
						|
 *     see forge.tls.CipherSuites. | 
						|
 *   virtualHost: the virtual server name to use in a TLS SNI | 
						|
 *     extension, if not provided the url host will be used. | 
						|
 *   verify: a custom TLS certificate verify callback to use. | 
						|
 *   getCertificate: an optional callback used to get a client-side | 
						|
 *     certificate (see forge.tls for details). | 
						|
 *   getPrivateKey: an optional callback used to get a client-side | 
						|
 *     private key (see forge.tls for details). | 
						|
 *   getSignature: an optional callback used to get a client-side | 
						|
 *     signature (see forge.tls for details). | 
						|
 *   persistCookies: true to use persistent cookies via flash local | 
						|
 *     storage, false to only keep cookies in javascript. | 
						|
 *   primeTlsSockets: true to immediately connect TLS sockets on | 
						|
 *     their creation so that they will cache TLS sessions for reuse. | 
						|
 * | 
						|
 * @return the client. | 
						|
 */ | 
						|
http.createClient = function(options) { | 
						|
  // create CA store to share with all TLS connections | 
						|
  var caStore = null; | 
						|
  if(options.caCerts) { | 
						|
    caStore = forge.pki.createCaStore(options.caCerts); | 
						|
  } | 
						|
 | 
						|
  // get scheme, host, and port from url | 
						|
  options.url = (options.url || | 
						|
    window.location.protocol + '//' + window.location.host); | 
						|
  var url = http.parseUrl(options.url); | 
						|
  if(!url) { | 
						|
    var error = new Error('Invalid url.'); | 
						|
    error.details = {url: options.url}; | 
						|
    throw error; | 
						|
  } | 
						|
 | 
						|
  // default to 1 connection | 
						|
  options.connections = options.connections || 1; | 
						|
 | 
						|
  // create client | 
						|
  var sp = options.socketPool; | 
						|
  var client = { | 
						|
    // url | 
						|
    url: url, | 
						|
    // socket pool | 
						|
    socketPool: sp, | 
						|
    // the policy port to use | 
						|
    policyPort: options.policyPort, | 
						|
    // policy url to use | 
						|
    policyUrl: options.policyUrl, | 
						|
    // queue of requests to service | 
						|
    requests: [], | 
						|
    // all sockets | 
						|
    sockets: [], | 
						|
    // idle sockets | 
						|
    idle: [], | 
						|
    // whether or not the connections are secure | 
						|
    secure: (url.scheme === 'https'), | 
						|
    // cookie jar (key'd off of name and then path, there is only 1 domain | 
						|
    // and one setting for secure per client so name+path is unique) | 
						|
    cookies: {}, | 
						|
    // default to flash storage of cookies | 
						|
    persistCookies: (typeof(options.persistCookies) === 'undefined') ? | 
						|
      true : options.persistCookies | 
						|
  }; | 
						|
 | 
						|
  // add client to debug storage | 
						|
  if(forge.debug) { | 
						|
    forge.debug.get('forge.http', 'clients').push(client); | 
						|
  } | 
						|
 | 
						|
  // load cookies from disk | 
						|
  _loadCookies(client); | 
						|
 | 
						|
  /** | 
						|
   * A default certificate verify function that checks a certificate common | 
						|
   * name against the client's URL host. | 
						|
   * | 
						|
   * @param c the TLS connection. | 
						|
   * @param verified true if cert is verified, otherwise alert number. | 
						|
   * @param depth the chain depth. | 
						|
   * @param certs the cert chain. | 
						|
   * | 
						|
   * @return true if verified and the common name matches the host, error | 
						|
   *         otherwise. | 
						|
   */ | 
						|
  var _defaultCertificateVerify = function(c, verified, depth, certs) { | 
						|
    if(depth === 0 && verified === true) { | 
						|
      // compare common name to url host | 
						|
      var cn = certs[depth].subject.getField('CN'); | 
						|
      if(cn === null || client.url.host !== cn.value) { | 
						|
        verified = { | 
						|
          message: 'Certificate common name does not match url host.' | 
						|
        }; | 
						|
      } | 
						|
    } | 
						|
    return verified; | 
						|
  }; | 
						|
 | 
						|
  // determine if TLS is used | 
						|
  var tlsOptions = null; | 
						|
  if(client.secure) { | 
						|
    tlsOptions = { | 
						|
      caStore: caStore, | 
						|
      cipherSuites: options.cipherSuites || null, | 
						|
      virtualHost: options.virtualHost || url.host, | 
						|
      verify: options.verify || _defaultCertificateVerify, | 
						|
      getCertificate: options.getCertificate || null, | 
						|
      getPrivateKey: options.getPrivateKey || null, | 
						|
      getSignature: options.getSignature || null, | 
						|
      prime: options.primeTlsSockets || false | 
						|
    }; | 
						|
 | 
						|
    // if socket pool uses a flash api, then add deflate support to TLS | 
						|
    if(sp.flashApi !== null) { | 
						|
      tlsOptions.deflate = function(bytes) { | 
						|
        // strip 2 byte zlib header and 4 byte trailer | 
						|
        return forge.util.deflate(sp.flashApi, bytes, true); | 
						|
      }; | 
						|
      tlsOptions.inflate = function(bytes) { | 
						|
        return forge.util.inflate(sp.flashApi, bytes, true); | 
						|
      }; | 
						|
    } | 
						|
  } | 
						|
 | 
						|
  // create and initialize sockets | 
						|
  for(var i = 0; i < options.connections; ++i) { | 
						|
    _initSocket(client, sp.createSocket(), tlsOptions); | 
						|
  } | 
						|
 | 
						|
  /** | 
						|
   * Sends a request. A method 'abort' will be set on the request that | 
						|
   * can be called to attempt to abort the request. | 
						|
   * | 
						|
   * @param options: | 
						|
   *          request: the request to send. | 
						|
   *          connected: a callback for when the connection is open. | 
						|
   *          closed: a callback for when the connection is closed. | 
						|
   *          headerReady: a callback for when the response header arrives. | 
						|
   *          bodyReady: a callback for when the response body arrives. | 
						|
   *          error: a callback for if an error occurs. | 
						|
   */ | 
						|
  client.send = function(options) { | 
						|
    // add host header if not set | 
						|
    if(options.request.getField('Host') === null) { | 
						|
      options.request.setField('Host', client.url.fullHost); | 
						|
    } | 
						|
 | 
						|
    // set default dummy handlers | 
						|
    var opts = {}; | 
						|
    opts.request = options.request; | 
						|
    opts.connected = options.connected || function() {}; | 
						|
    opts.closed = options.close || function() {}; | 
						|
    opts.headerReady = function(e) { | 
						|
      // read cookies | 
						|
      _readCookies(client, e.response); | 
						|
      if(options.headerReady) { | 
						|
        options.headerReady(e); | 
						|
      } | 
						|
    }; | 
						|
    opts.bodyReady = options.bodyReady || function() {}; | 
						|
    opts.error = options.error || function() {}; | 
						|
 | 
						|
    // create response | 
						|
    opts.response = http.createResponse(); | 
						|
    opts.response.time = 0; | 
						|
    opts.response.flashApi = client.socketPool.flashApi; | 
						|
    opts.request.flashApi = client.socketPool.flashApi; | 
						|
 | 
						|
    // create abort function | 
						|
    opts.request.abort = function() { | 
						|
      // set aborted, clear handlers | 
						|
      opts.request.aborted = true; | 
						|
      opts.connected = function() {}; | 
						|
      opts.closed = function() {}; | 
						|
      opts.headerReady = function() {}; | 
						|
      opts.bodyReady = function() {}; | 
						|
      opts.error = function() {}; | 
						|
    }; | 
						|
 | 
						|
    // add cookies to request | 
						|
    _writeCookies(client, opts.request); | 
						|
 | 
						|
    // queue request options if there are no idle sockets | 
						|
    if(client.idle.length === 0) { | 
						|
      client.requests.push(opts); | 
						|
    } else { | 
						|
      // use an idle socket, prefer an idle *connected* socket first | 
						|
      var socket = null; | 
						|
      var len = client.idle.length; | 
						|
      for(var i = 0; socket === null && i < len; ++i) { | 
						|
        socket = client.idle[i]; | 
						|
        if(socket.isConnected()) { | 
						|
          client.idle.splice(i, 1); | 
						|
        } else { | 
						|
          socket = null; | 
						|
        } | 
						|
      } | 
						|
      // no connected socket available, get unconnected socket | 
						|
      if(socket === null) { | 
						|
        socket = client.idle.pop(); | 
						|
      } | 
						|
      socket.options = opts; | 
						|
      _doRequest(client, socket); | 
						|
    } | 
						|
  }; | 
						|
 | 
						|
  /** | 
						|
   * Destroys this client. | 
						|
   */ | 
						|
  client.destroy = function() { | 
						|
    // clear pending requests, close and destroy sockets | 
						|
    client.requests = []; | 
						|
    for(var i = 0; i < client.sockets.length; ++i) { | 
						|
      client.sockets[i].close(); | 
						|
      client.sockets[i].destroy(); | 
						|
    } | 
						|
    client.socketPool = null; | 
						|
    client.sockets = []; | 
						|
    client.idle = []; | 
						|
  }; | 
						|
 | 
						|
  /** | 
						|
   * Sets a cookie for use with all connections made by this client. Any | 
						|
   * cookie with the same name will be replaced. If the cookie's value | 
						|
   * is undefined, null, or the blank string, the cookie will be removed. | 
						|
   * | 
						|
   * If the cookie's domain doesn't match this client's url host or the | 
						|
   * cookie's secure flag doesn't match this client's url scheme, then | 
						|
   * setting the cookie will fail with an exception. | 
						|
   * | 
						|
   * @param cookie the cookie with parameters: | 
						|
   *   name: the name of the cookie. | 
						|
   *   value: the value of the cookie. | 
						|
   *   comment: an optional comment string. | 
						|
   *   maxAge: the age of the cookie in seconds relative to created time. | 
						|
   *   secure: true if the cookie must be sent over a secure protocol. | 
						|
   *   httpOnly: true to restrict access to the cookie from javascript | 
						|
   *     (inaffective since the cookies are stored in javascript). | 
						|
   *   path: the path for the cookie. | 
						|
   *   domain: optional domain the cookie belongs to (must start with dot). | 
						|
   *   version: optional version of the cookie. | 
						|
   *   created: creation time, in UTC seconds, of the cookie. | 
						|
   */ | 
						|
  client.setCookie = function(cookie) { | 
						|
    var rval; | 
						|
    if(typeof(cookie.name) !== 'undefined') { | 
						|
      if(cookie.value === null || typeof(cookie.value) === 'undefined' || | 
						|
        cookie.value === '') { | 
						|
        // remove cookie | 
						|
        rval = client.removeCookie(cookie.name, cookie.path); | 
						|
      } else { | 
						|
        // set cookie defaults | 
						|
        cookie.comment = cookie.comment || ''; | 
						|
        cookie.maxAge = cookie.maxAge || 0; | 
						|
        cookie.secure = (typeof(cookie.secure) === 'undefined') ? | 
						|
          true : cookie.secure; | 
						|
        cookie.httpOnly = cookie.httpOnly || true; | 
						|
        cookie.path = cookie.path || '/'; | 
						|
        cookie.domain = cookie.domain || null; | 
						|
        cookie.version = cookie.version || null; | 
						|
        cookie.created = _getUtcTime(new Date()); | 
						|
 | 
						|
        // do secure check | 
						|
        if(cookie.secure !== client.secure) { | 
						|
          var error = new Error('Http client url scheme is incompatible ' + | 
						|
            'with cookie secure flag.'); | 
						|
          error.url = client.url; | 
						|
          error.cookie = cookie; | 
						|
          throw error; | 
						|
        } | 
						|
        // make sure url host is within cookie.domain | 
						|
        if(!http.withinCookieDomain(client.url, cookie)) { | 
						|
          var error = new Error('Http client url scheme is incompatible ' + | 
						|
            'with cookie secure flag.'); | 
						|
          error.url = client.url; | 
						|
          error.cookie = cookie; | 
						|
          throw error; | 
						|
        } | 
						|
 | 
						|
        // add new cookie | 
						|
        if(!(cookie.name in client.cookies)) { | 
						|
          client.cookies[cookie.name] = {}; | 
						|
        } | 
						|
        client.cookies[cookie.name][cookie.path] = cookie; | 
						|
        rval = true; | 
						|
 | 
						|
        // save cookies | 
						|
        _saveCookies(client); | 
						|
      } | 
						|
    } | 
						|
 | 
						|
    return rval; | 
						|
  }; | 
						|
 | 
						|
  /** | 
						|
   * Gets a cookie by its name. | 
						|
   * | 
						|
   * @param name the name of the cookie to retrieve. | 
						|
   * @param path an optional path for the cookie (if there are multiple | 
						|
   *          cookies with the same name but different paths). | 
						|
   * | 
						|
   * @return the cookie or null if not found. | 
						|
   */ | 
						|
  client.getCookie = function(name, path) { | 
						|
    var rval = null; | 
						|
    if(name in client.cookies) { | 
						|
      var paths = client.cookies[name]; | 
						|
 | 
						|
      // get path-specific cookie | 
						|
      if(path) { | 
						|
        if(path in paths) { | 
						|
          rval = paths[path]; | 
						|
        } | 
						|
      } else { | 
						|
        // get first cookie | 
						|
        for(var p in paths) { | 
						|
          rval = paths[p]; | 
						|
          break; | 
						|
        } | 
						|
      } | 
						|
    } | 
						|
    return rval; | 
						|
  }; | 
						|
 | 
						|
  /** | 
						|
   * Removes a cookie. | 
						|
   * | 
						|
   * @param name the name of the cookie to remove. | 
						|
   * @param path an optional path for the cookie (if there are multiple | 
						|
   *          cookies with the same name but different paths). | 
						|
   * | 
						|
   * @return true if a cookie was removed, false if not. | 
						|
   */ | 
						|
  client.removeCookie = function(name, path) { | 
						|
    var rval = false; | 
						|
    if(name in client.cookies) { | 
						|
      // delete the specific path | 
						|
      if(path) { | 
						|
        var paths = client.cookies[name]; | 
						|
        if(path in paths) { | 
						|
          rval = true; | 
						|
          delete client.cookies[name][path]; | 
						|
          // clean up entry if empty | 
						|
          var empty = true; | 
						|
          for(var i in client.cookies[name]) { | 
						|
            empty = false; | 
						|
            break; | 
						|
          } | 
						|
          if(empty) { | 
						|
            delete client.cookies[name]; | 
						|
          } | 
						|
        } | 
						|
      } else { | 
						|
        // delete all cookies with the given name | 
						|
        rval = true; | 
						|
        delete client.cookies[name]; | 
						|
      } | 
						|
    } | 
						|
    if(rval) { | 
						|
      // save cookies | 
						|
      _saveCookies(client); | 
						|
    } | 
						|
    return rval; | 
						|
  }; | 
						|
 | 
						|
  /** | 
						|
   * Clears all cookies stored in this client. | 
						|
   */ | 
						|
  client.clearCookies = function() { | 
						|
    client.cookies = {}; | 
						|
    _clearCookies(client); | 
						|
  }; | 
						|
 | 
						|
  if(forge.log) { | 
						|
    forge.log.debug('forge.http', 'created client', options); | 
						|
  } | 
						|
 | 
						|
  return client; | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Trims the whitespace off of the beginning and end of a string. | 
						|
 * | 
						|
 * @param str the string to trim. | 
						|
 * | 
						|
 * @return the trimmed string. | 
						|
 */ | 
						|
var _trimString = function(str) { | 
						|
  return str.replace(/^\s*/, '').replace(/\s*$/, ''); | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Creates an http header object. | 
						|
 * | 
						|
 * @return the http header object. | 
						|
 */ | 
						|
var _createHeader = function() { | 
						|
  var header = { | 
						|
    fields: {}, | 
						|
    setField: function(name, value) { | 
						|
      // normalize field name, trim value | 
						|
      header.fields[_normalize(name)] = [_trimString('' + value)]; | 
						|
    }, | 
						|
    appendField: function(name, value) { | 
						|
      name = _normalize(name); | 
						|
      if(!(name in header.fields)) { | 
						|
        header.fields[name] = []; | 
						|
      } | 
						|
      header.fields[name].push(_trimString('' + value)); | 
						|
    }, | 
						|
    getField: function(name, index) { | 
						|
      var rval = null; | 
						|
      name = _normalize(name); | 
						|
      if(name in header.fields) { | 
						|
        index = index || 0; | 
						|
        rval = header.fields[name][index]; | 
						|
      } | 
						|
      return rval; | 
						|
    } | 
						|
  }; | 
						|
  return header; | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Gets the time in utc seconds given a date. | 
						|
 * | 
						|
 * @param d the date to use. | 
						|
 * | 
						|
 * @return the time in utc seconds. | 
						|
 */ | 
						|
var _getUtcTime = function(d) { | 
						|
  var utc = +d + d.getTimezoneOffset() * 60000; | 
						|
  return Math.floor(+new Date() / 1000); | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Creates an http request. | 
						|
 * | 
						|
 * @param options: | 
						|
 *          version: the version. | 
						|
 *          method: the method. | 
						|
 *          path: the path. | 
						|
 *          body: the body. | 
						|
 *          headers: custom header fields to add, | 
						|
 *            eg: [{'Content-Length': 0}]. | 
						|
 * | 
						|
 * @return the http request. | 
						|
 */ | 
						|
http.createRequest = function(options) { | 
						|
  options = options || {}; | 
						|
  var request = _createHeader(); | 
						|
  request.version = options.version || 'HTTP/1.1'; | 
						|
  request.method = options.method || null; | 
						|
  request.path = options.path || null; | 
						|
  request.body = options.body || null; | 
						|
  request.bodyDeflated = false; | 
						|
  request.flashApi = null; | 
						|
 | 
						|
  // add custom headers | 
						|
  var headers = options.headers || []; | 
						|
  if(!forge.util.isArray(headers)) { | 
						|
    headers = [headers]; | 
						|
  } | 
						|
  for(var i = 0; i < headers.length; ++i) { | 
						|
    for(var name in headers[i]) { | 
						|
      request.appendField(name, headers[i][name]); | 
						|
    } | 
						|
  } | 
						|
 | 
						|
  /** | 
						|
   * Adds a cookie to the request 'Cookie' header. | 
						|
   * | 
						|
   * @param cookie a cookie to add. | 
						|
   */ | 
						|
  request.addCookie = function(cookie) { | 
						|
    var value = ''; | 
						|
    var field = request.getField('Cookie'); | 
						|
    if(field !== null) { | 
						|
      // separate cookies by semi-colons | 
						|
      value = field + '; '; | 
						|
    } | 
						|
 | 
						|
    // get current time in utc seconds | 
						|
    var now = _getUtcTime(new Date()); | 
						|
 | 
						|
    // output cookie name and value | 
						|
    value += cookie.name + '=' + cookie.value; | 
						|
    request.setField('Cookie', value); | 
						|
  }; | 
						|
 | 
						|
  /** | 
						|
   * Converts an http request into a string that can be sent as an | 
						|
   * HTTP request. Does not include any data. | 
						|
   * | 
						|
   * @return the string representation of the request. | 
						|
   */ | 
						|
  request.toString = function() { | 
						|
    /* Sample request header: | 
						|
      GET /some/path/?query HTTP/1.1 | 
						|
      Host: www.someurl.com | 
						|
      Connection: close | 
						|
      Accept-Encoding: deflate | 
						|
      Accept: image/gif, text/html | 
						|
      User-Agent: Mozilla 4.0 | 
						|
     */ | 
						|
 | 
						|
    // set default headers | 
						|
    if(request.getField('User-Agent') === null) { | 
						|
      request.setField('User-Agent', 'forge.http 1.0'); | 
						|
    } | 
						|
    if(request.getField('Accept') === null) { | 
						|
      request.setField('Accept', '*/*'); | 
						|
    } | 
						|
    if(request.getField('Connection') === null) { | 
						|
      request.setField('Connection', 'keep-alive'); | 
						|
      request.setField('Keep-Alive', '115'); | 
						|
    } | 
						|
 | 
						|
    // add Accept-Encoding if not specified | 
						|
    if(request.flashApi !== null && | 
						|
      request.getField('Accept-Encoding') === null) { | 
						|
      request.setField('Accept-Encoding', 'deflate'); | 
						|
    } | 
						|
 | 
						|
    // if the body isn't null, deflate it if its larger than 100 bytes | 
						|
    if(request.flashApi !== null && request.body !== null && | 
						|
      request.getField('Content-Encoding') === null && | 
						|
      !request.bodyDeflated && request.body.length > 100) { | 
						|
      // use flash to compress data | 
						|
      request.body = forge.util.deflate(request.flashApi, request.body); | 
						|
      request.bodyDeflated = true; | 
						|
      request.setField('Content-Encoding', 'deflate'); | 
						|
      request.setField('Content-Length', request.body.length); | 
						|
    } else if(request.body !== null) { | 
						|
      // set content length for body | 
						|
      request.setField('Content-Length', request.body.length); | 
						|
    } | 
						|
 | 
						|
    // build start line | 
						|
    var rval = | 
						|
      request.method.toUpperCase() + ' ' + request.path + ' ' + | 
						|
      request.version + '\r\n'; | 
						|
 | 
						|
    // add each header | 
						|
    for(var name in request.fields) { | 
						|
      var fields = request.fields[name]; | 
						|
      for(var i = 0; i < fields.length; ++i) { | 
						|
        rval += name + ': ' + fields[i] + '\r\n'; | 
						|
      } | 
						|
    } | 
						|
    // final terminating CRLF | 
						|
    rval += '\r\n'; | 
						|
 | 
						|
    return rval; | 
						|
  }; | 
						|
 | 
						|
  return request; | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Creates an empty http response header. | 
						|
 * | 
						|
 * @return the empty http response header. | 
						|
 */ | 
						|
http.createResponse = function() { | 
						|
  // private vars | 
						|
  var _first = true; | 
						|
  var _chunkSize = 0; | 
						|
  var _chunksFinished = false; | 
						|
 | 
						|
  // create response | 
						|
  var response = _createHeader(); | 
						|
  response.version = null; | 
						|
  response.code = 0; | 
						|
  response.message = null; | 
						|
  response.body = null; | 
						|
  response.headerReceived = false; | 
						|
  response.bodyReceived = false; | 
						|
  response.flashApi = null; | 
						|
 | 
						|
  /** | 
						|
   * Reads a line that ends in CRLF from a byte buffer. | 
						|
   * | 
						|
   * @param b the byte buffer. | 
						|
   * | 
						|
   * @return the line or null if none was found. | 
						|
   */ | 
						|
  var _readCrlf = function(b) { | 
						|
    var line = null; | 
						|
    var i = b.data.indexOf('\r\n', b.read); | 
						|
    if(i != -1) { | 
						|
      // read line, skip CRLF | 
						|
      line = b.getBytes(i - b.read); | 
						|
      b.getBytes(2); | 
						|
    } | 
						|
    return line; | 
						|
  }; | 
						|
 | 
						|
  /** | 
						|
   * Parses a header field and appends it to the response. | 
						|
   * | 
						|
   * @param line the header field line. | 
						|
   */ | 
						|
  var _parseHeader = function(line) { | 
						|
    var tmp = line.indexOf(':'); | 
						|
    var name = line.substring(0, tmp++); | 
						|
    response.appendField( | 
						|
      name, (tmp < line.length) ? line.substring(tmp) : ''); | 
						|
  }; | 
						|
 | 
						|
  /** | 
						|
   * Reads an http response header from a buffer of bytes. | 
						|
   * | 
						|
   * @param b the byte buffer to parse the header from. | 
						|
   * | 
						|
   * @return true if the whole header was read, false if not. | 
						|
   */ | 
						|
  response.readHeader = function(b) { | 
						|
    // read header lines (each ends in CRLF) | 
						|
    var line = ''; | 
						|
    while(!response.headerReceived && line !== null) { | 
						|
      line = _readCrlf(b); | 
						|
      if(line !== null) { | 
						|
        // parse first line | 
						|
        if(_first) { | 
						|
          _first = false; | 
						|
          var tmp = line.split(' '); | 
						|
          if(tmp.length >= 3) { | 
						|
            response.version = tmp[0]; | 
						|
            response.code = parseInt(tmp[1], 10); | 
						|
            response.message = tmp.slice(2).join(' '); | 
						|
          } else { | 
						|
            // invalid header | 
						|
            var error = new Error('Invalid http response header.'); | 
						|
            error.details = {'line': line}; | 
						|
            throw error; | 
						|
          } | 
						|
        } else if(line.length === 0) { | 
						|
          // handle final line, end of header | 
						|
          response.headerReceived = true; | 
						|
        } else { | 
						|
          _parseHeader(line); | 
						|
        } | 
						|
      } | 
						|
    } | 
						|
 | 
						|
    return response.headerReceived; | 
						|
  }; | 
						|
 | 
						|
  /** | 
						|
   * Reads some chunked http response entity-body from the given buffer of | 
						|
   * bytes. | 
						|
   * | 
						|
   * @param b the byte buffer to read from. | 
						|
   * | 
						|
   * @return true if the whole body was read, false if not. | 
						|
   */ | 
						|
  var _readChunkedBody = function(b) { | 
						|
    /* Chunked transfer-encoding sends data in a series of chunks, | 
						|
      followed by a set of 0-N http trailers. | 
						|
      The format is as follows: | 
						|
 | 
						|
      chunk-size (in hex) CRLF | 
						|
      chunk data (with "chunk-size" many bytes) CRLF | 
						|
      ... (N many chunks) | 
						|
      chunk-size (of 0 indicating the last chunk) CRLF | 
						|
      N many http trailers followed by CRLF | 
						|
      blank line + CRLF (terminates the trailers) | 
						|
 | 
						|
      If there are no http trailers, then after the chunk-size of 0, | 
						|
      there is still a single CRLF (indicating the blank line + CRLF | 
						|
      that terminates the trailers). In other words, you always terminate | 
						|
      the trailers with blank line + CRLF, regardless of 0-N trailers. */ | 
						|
 | 
						|
      /* From RFC-2616, section 3.6.1, here is the pseudo-code for | 
						|
      implementing chunked transfer-encoding: | 
						|
 | 
						|
      length := 0 | 
						|
      read chunk-size, chunk-extension (if any) and CRLF | 
						|
      while (chunk-size > 0) { | 
						|
        read chunk-data and CRLF | 
						|
        append chunk-data to entity-body | 
						|
        length := length + chunk-size | 
						|
        read chunk-size and CRLF | 
						|
      } | 
						|
      read entity-header | 
						|
      while (entity-header not empty) { | 
						|
        append entity-header to existing header fields | 
						|
        read entity-header | 
						|
      } | 
						|
      Content-Length := length | 
						|
      Remove "chunked" from Transfer-Encoding | 
						|
    */ | 
						|
 | 
						|
    var line = ''; | 
						|
    while(line !== null && b.length() > 0) { | 
						|
      // if in the process of reading a chunk | 
						|
      if(_chunkSize > 0) { | 
						|
        // if there are not enough bytes to read chunk and its | 
						|
        // trailing CRLF,  we must wait for more data to be received | 
						|
        if(_chunkSize + 2 > b.length()) { | 
						|
          break; | 
						|
        } | 
						|
 | 
						|
        // read chunk data, skip CRLF | 
						|
        response.body += b.getBytes(_chunkSize); | 
						|
        b.getBytes(2); | 
						|
        _chunkSize = 0; | 
						|
      } else if(!_chunksFinished) { | 
						|
        // more chunks, read next chunk-size line | 
						|
        line = _readCrlf(b); | 
						|
        if(line !== null) { | 
						|
          // parse chunk-size (ignore any chunk extension) | 
						|
          _chunkSize = parseInt(line.split(';', 1)[0], 16); | 
						|
          _chunksFinished = (_chunkSize === 0); | 
						|
        } | 
						|
      } else { | 
						|
        // chunks finished, read next trailer | 
						|
        line = _readCrlf(b); | 
						|
        while(line !== null) { | 
						|
          if(line.length > 0) { | 
						|
            // parse trailer | 
						|
            _parseHeader(line); | 
						|
            // read next trailer | 
						|
            line = _readCrlf(b); | 
						|
          } else { | 
						|
            // body received | 
						|
            response.bodyReceived = true; | 
						|
            line = null; | 
						|
          } | 
						|
        } | 
						|
      } | 
						|
    } | 
						|
 | 
						|
    return response.bodyReceived; | 
						|
  }; | 
						|
 | 
						|
  /** | 
						|
   * Reads an http response body from a buffer of bytes. | 
						|
   * | 
						|
   * @param b the byte buffer to read from. | 
						|
   * | 
						|
   * @return true if the whole body was read, false if not. | 
						|
   */ | 
						|
  response.readBody = function(b) { | 
						|
    var contentLength = response.getField('Content-Length'); | 
						|
    var transferEncoding = response.getField('Transfer-Encoding'); | 
						|
    if(contentLength !== null) { | 
						|
      contentLength = parseInt(contentLength); | 
						|
    } | 
						|
 | 
						|
    // read specified length | 
						|
    if(contentLength !== null && contentLength >= 0) { | 
						|
      response.body = response.body || ''; | 
						|
      response.body += b.getBytes(contentLength); | 
						|
      response.bodyReceived = (response.body.length === contentLength); | 
						|
    } else if(transferEncoding !== null) { | 
						|
      // read chunked encoding | 
						|
      if(transferEncoding.indexOf('chunked') != -1) { | 
						|
        response.body = response.body || ''; | 
						|
        _readChunkedBody(b); | 
						|
      } else { | 
						|
        var error = new Error('Unknown Transfer-Encoding.'); | 
						|
        error.details = {'transferEncoding': transferEncoding}; | 
						|
        throw error; | 
						|
      } | 
						|
    } else if((contentLength !== null && contentLength < 0) || | 
						|
      (contentLength === null && | 
						|
      response.getField('Content-Type') !== null)) { | 
						|
      // read all data in the buffer | 
						|
      response.body = response.body || ''; | 
						|
      response.body += b.getBytes(); | 
						|
      response.readBodyUntilClose = true; | 
						|
    } else { | 
						|
      // no body | 
						|
      response.body = null; | 
						|
      response.bodyReceived = true; | 
						|
    } | 
						|
 | 
						|
    if(response.bodyReceived) { | 
						|
      response.time = +new Date() - response.time; | 
						|
    } | 
						|
 | 
						|
    if(response.flashApi !== null && | 
						|
      response.bodyReceived && response.body !== null && | 
						|
      response.getField('Content-Encoding') === 'deflate') { | 
						|
      // inflate using flash api | 
						|
      response.body = forge.util.inflate( | 
						|
        response.flashApi, response.body); | 
						|
    } | 
						|
 | 
						|
    return response.bodyReceived; | 
						|
  }; | 
						|
 | 
						|
   /** | 
						|
    * Parses an array of cookies from the 'Set-Cookie' field, if present. | 
						|
    * | 
						|
    * @return the array of cookies. | 
						|
    */ | 
						|
   response.getCookies = function() { | 
						|
     var rval = []; | 
						|
 | 
						|
     // get Set-Cookie field | 
						|
     if('Set-Cookie' in response.fields) { | 
						|
       var field = response.fields['Set-Cookie']; | 
						|
 | 
						|
       // get current local time in seconds | 
						|
       var now = +new Date() / 1000; | 
						|
 | 
						|
       // regex for parsing 'name1=value1; name2=value2; name3' | 
						|
       var regex = /\s*([^=]*)=?([^;]*)(;|$)/g; | 
						|
 | 
						|
       // examples: | 
						|
       // Set-Cookie: cookie1_name=cookie1_value; max-age=0; path=/ | 
						|
       // Set-Cookie: c2=v2; expires=Thu, 21-Aug-2008 23:47:25 GMT; path=/ | 
						|
       for(var i = 0; i < field.length; ++i) { | 
						|
         var fv = field[i]; | 
						|
         var m; | 
						|
         regex.lastIndex = 0; | 
						|
         var first = true; | 
						|
         var cookie = {}; | 
						|
         do { | 
						|
           m = regex.exec(fv); | 
						|
           if(m !== null) { | 
						|
             var name = _trimString(m[1]); | 
						|
             var value = _trimString(m[2]); | 
						|
 | 
						|
             // cookie_name=value | 
						|
             if(first) { | 
						|
               cookie.name = name; | 
						|
               cookie.value = value; | 
						|
               first = false; | 
						|
             } else { | 
						|
               // property_name=value | 
						|
               name = name.toLowerCase(); | 
						|
               switch(name) { | 
						|
               case 'expires': | 
						|
                 // replace hyphens w/spaces so date will parse | 
						|
                 value = value.replace(/-/g, ' '); | 
						|
                 var secs = Date.parse(value) / 1000; | 
						|
                 cookie.maxAge = Math.max(0, secs - now); | 
						|
                 break; | 
						|
               case 'max-age': | 
						|
                 cookie.maxAge = parseInt(value, 10); | 
						|
                 break; | 
						|
               case 'secure': | 
						|
                 cookie.secure = true; | 
						|
                 break; | 
						|
               case 'httponly': | 
						|
                 cookie.httpOnly = true; | 
						|
                 break; | 
						|
               default: | 
						|
                 if(name !== '') { | 
						|
                   cookie[name] = value; | 
						|
                 } | 
						|
               } | 
						|
             } | 
						|
           } | 
						|
         } while(m !== null && m[0] !== ''); | 
						|
         rval.push(cookie); | 
						|
       } | 
						|
     } | 
						|
 | 
						|
     return rval; | 
						|
  }; | 
						|
 | 
						|
  /** | 
						|
   * Converts an http response into a string that can be sent as an | 
						|
   * HTTP response. Does not include any data. | 
						|
   * | 
						|
   * @return the string representation of the response. | 
						|
   */ | 
						|
  response.toString = function() { | 
						|
    /* Sample response header: | 
						|
      HTTP/1.0 200 OK | 
						|
      Host: www.someurl.com | 
						|
      Connection: close | 
						|
     */ | 
						|
 | 
						|
    // build start line | 
						|
    var rval = | 
						|
      response.version + ' ' + response.code + ' ' + response.message + '\r\n'; | 
						|
 | 
						|
    // add each header | 
						|
    for(var name in response.fields) { | 
						|
      var fields = response.fields[name]; | 
						|
      for(var i = 0; i < fields.length; ++i) { | 
						|
        rval += name + ': ' + fields[i] + '\r\n'; | 
						|
      } | 
						|
    } | 
						|
    // final terminating CRLF | 
						|
    rval += '\r\n'; | 
						|
 | 
						|
    return rval; | 
						|
  }; | 
						|
 | 
						|
  return response; | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Parses the scheme, host, and port from an http(s) url. | 
						|
 * | 
						|
 * @param str the url string. | 
						|
 * | 
						|
 * @return the parsed url object or null if the url is invalid. | 
						|
 */ | 
						|
http.parseUrl = forge.util.parseUrl; | 
						|
 | 
						|
/** | 
						|
 * Returns true if the given url is within the given cookie's domain. | 
						|
 * | 
						|
 * @param url the url to check. | 
						|
 * @param cookie the cookie or cookie domain to check. | 
						|
 */ | 
						|
http.withinCookieDomain = function(url, cookie) { | 
						|
  var rval = false; | 
						|
 | 
						|
  // cookie may be null, a cookie object, or a domain string | 
						|
  var domain = (cookie === null || typeof cookie === 'string') ? | 
						|
    cookie : cookie.domain; | 
						|
 | 
						|
  // any domain will do | 
						|
  if(domain === null) { | 
						|
    rval = true; | 
						|
  } else if(domain.charAt(0) === '.') { | 
						|
    // ensure domain starts with a '.' | 
						|
    // parse URL as necessary | 
						|
    if(typeof url === 'string') { | 
						|
      url = http.parseUrl(url); | 
						|
    } | 
						|
 | 
						|
    // add '.' to front of URL host to match against domain | 
						|
    var host = '.' + url.host; | 
						|
 | 
						|
    // if the host ends with domain then it falls within it | 
						|
    var idx = host.lastIndexOf(domain); | 
						|
    if(idx !== -1 && (idx + domain.length === host.length)) { | 
						|
      rval = true; | 
						|
    } | 
						|
  } | 
						|
 | 
						|
  return rval; | 
						|
};
 | 
						|
 |