255 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
			
		
		
	
	
			255 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
/**
 | 
						|
 * Wrapper for built-in http.js to emulate the browser XMLHttpRequest object.
 | 
						|
 *
 | 
						|
 * This can be used with JS designed for browsers to improve reuse of code and
 | 
						|
 * allow the use of existing libraries.
 | 
						|
 *
 | 
						|
 * Usage: include("XMLHttpRequest.js") and use XMLHttpRequest per W3C specs.
 | 
						|
 *
 | 
						|
 * @todo SSL Support
 | 
						|
 * @author Dan DeFelippi <dan@driverdan.com>
 | 
						|
 * @license MIT
 | 
						|
 */
 | 
						|
 | 
						|
var Url = require("url")
 | 
						|
	,sys = require("util");
 | 
						|
 | 
						|
exports.XMLHttpRequest = function() {
 | 
						|
	/**
 | 
						|
	 * Private variables
 | 
						|
	 */
 | 
						|
	var self = this;
 | 
						|
	var http = require('http');
 | 
						|
	var https = require('https');
 | 
						|
 | 
						|
	// Holds http.js objects
 | 
						|
	var client;
 | 
						|
	var request;
 | 
						|
	var response;
 | 
						|
	
 | 
						|
	// Request settings
 | 
						|
	var settings = {};
 | 
						|
	
 | 
						|
	// Set some default headers
 | 
						|
	var defaultHeaders = {
 | 
						|
		"User-Agent": "node.js",
 | 
						|
		"Accept": "*/*",
 | 
						|
	};
 | 
						|
	
 | 
						|
	var headers = defaultHeaders;
 | 
						|
	
 | 
						|
	/**
 | 
						|
	 * Constants
 | 
						|
	 */
 | 
						|
	this.UNSENT = 0;
 | 
						|
	this.OPENED = 1;
 | 
						|
	this.HEADERS_RECEIVED = 2;
 | 
						|
	this.LOADING = 3;
 | 
						|
	this.DONE = 4;
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Public vars
 | 
						|
	 */
 | 
						|
	// Current state
 | 
						|
	this.readyState = this.UNSENT;
 | 
						|
 | 
						|
	// default ready state change handler in case one is not set or is set late
 | 
						|
	this.onreadystatechange = function() {};
 | 
						|
 | 
						|
	// Result & response
 | 
						|
	this.responseText = "";
 | 
						|
	this.responseXML = "";
 | 
						|
	this.status = null;
 | 
						|
	this.statusText = null;
 | 
						|
		
 | 
						|
	/**
 | 
						|
	 * Open the connection. Currently supports local server requests.
 | 
						|
	 *
 | 
						|
	 * @param string method Connection method (eg GET, POST)
 | 
						|
	 * @param string url URL for the connection.
 | 
						|
	 * @param boolean async Asynchronous connection. Default is true.
 | 
						|
	 * @param string user Username for basic authentication (optional)
 | 
						|
	 * @param string password Password for basic authentication (optional)
 | 
						|
	 */
 | 
						|
	this.open = function(method, url, async, user, password) {
 | 
						|
		settings = {
 | 
						|
			"method": method,
 | 
						|
			"url": url,
 | 
						|
			"async": async || null,
 | 
						|
			"user": user || null,
 | 
						|
			"password": password || null
 | 
						|
		};
 | 
						|
		
 | 
						|
		this.abort();
 | 
						|
 | 
						|
		setState(this.OPENED);
 | 
						|
	};
 | 
						|
	
 | 
						|
	/**
 | 
						|
	 * Sets a header for the request.
 | 
						|
	 *
 | 
						|
	 * @param string header Header name
 | 
						|
	 * @param string value Header value
 | 
						|
	 */
 | 
						|
	this.setRequestHeader = function(header, value) {
 | 
						|
		headers[header] = value;
 | 
						|
	};
 | 
						|
	
 | 
						|
	/**
 | 
						|
	 * Gets a header from the server response.
 | 
						|
	 *
 | 
						|
	 * @param string header Name of header to get.
 | 
						|
	 * @return string Text of the header or null if it doesn't exist.
 | 
						|
	 */
 | 
						|
	this.getResponseHeader = function(header) {
 | 
						|
		if (this.readyState > this.OPENED && response.headers[header]) {
 | 
						|
			return header + ": " + response.headers[header];
 | 
						|
		}
 | 
						|
		
 | 
						|
		return null;
 | 
						|
	};
 | 
						|
	
 | 
						|
	/**
 | 
						|
	 * Gets all the response headers.
 | 
						|
	 *
 | 
						|
	 * @return string 
 | 
						|
	 */
 | 
						|
	this.getAllResponseHeaders = function() {
 | 
						|
		if (this.readyState < this.HEADERS_RECEIVED) {
 | 
						|
			throw "INVALID_STATE_ERR: Headers have not been received.";
 | 
						|
		}
 | 
						|
		var result = "";
 | 
						|
		
 | 
						|
		for (var i in response.headers) {
 | 
						|
			result += i + ": " + response.headers[i] + "\r\n";
 | 
						|
		}
 | 
						|
		return result.substr(0, result.length - 2);
 | 
						|
	};
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Sends the request to the server.
 | 
						|
	 *
 | 
						|
	 * @param string data Optional data to send as request body.
 | 
						|
	 */
 | 
						|
	this.send = function(data) {
 | 
						|
		if (this.readyState != this.OPENED) {
 | 
						|
			throw "INVALID_STATE_ERR: connection must be opened before send() is called";
 | 
						|
		}
 | 
						|
		
 | 
						|
		var ssl = false;
 | 
						|
		var url = Url.parse(settings.url);
 | 
						|
		
 | 
						|
		// Determine the server
 | 
						|
		switch (url.protocol) {
 | 
						|
			case 'https:':
 | 
						|
				ssl = true;
 | 
						|
				// SSL & non-SSL both need host, no break here.
 | 
						|
			case 'http:':
 | 
						|
				var host = url.hostname;
 | 
						|
				break;
 | 
						|
			
 | 
						|
			case undefined:
 | 
						|
			case '':
 | 
						|
				var host = "localhost";
 | 
						|
				break;
 | 
						|
			
 | 
						|
			default:
 | 
						|
				throw "Protocol not supported.";
 | 
						|
		}
 | 
						|
 | 
						|
		// Default to port 80. If accessing localhost on another port be sure
 | 
						|
		// to use http://localhost:port/path
 | 
						|
		var port = url.port || (ssl ? 443 : 80);
 | 
						|
		// Add query string if one is used
 | 
						|
		var uri = url.pathname + (url.search ? url.search : '');
 | 
						|
		
 | 
						|
		// Set the Host header or the server may reject the request
 | 
						|
		this.setRequestHeader("Host", host);
 | 
						|
		
 | 
						|
		// Set content length header
 | 
						|
		if (settings.method == "GET" || settings.method == "HEAD") {
 | 
						|
			data = null;
 | 
						|
		} else if (data) {
 | 
						|
			this.setRequestHeader("Content-Length", Buffer.byteLength(data));
 | 
						|
			
 | 
						|
			if (!headers["Content-Type"]) {
 | 
						|
				this.setRequestHeader("Content-Type", "text/plain;charset=UTF-8");
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		// Use the proper protocol
 | 
						|
		var doRequest = ssl ? https.request : http.request;
 | 
						|
 | 
						|
		var options = {
 | 
						|
		    host: host,
 | 
						|
		    port: port,
 | 
						|
		    path: uri,
 | 
						|
		    method: settings.method,
 | 
						|
		    headers: headers, 
 | 
						|
                    agent: false
 | 
						|
		};
 | 
						|
		
 | 
						|
		var req = doRequest(options, function(res) {
 | 
						|
			response = res;
 | 
						|
			response.setEncoding("utf8");
 | 
						|
 | 
						|
			setState(self.HEADERS_RECEIVED);
 | 
						|
			self.status = response.statusCode;
 | 
						|
 | 
						|
			response.on('data', function(chunk) {
 | 
						|
				// Make sure there's some data
 | 
						|
				if (chunk) {
 | 
						|
					self.responseText += chunk;
 | 
						|
				}
 | 
						|
				setState(self.LOADING);
 | 
						|
			});
 | 
						|
 | 
						|
			response.on('end', function() {
 | 
						|
				setState(self.DONE);
 | 
						|
			});
 | 
						|
 | 
						|
			response.on('error', function() {
 | 
						|
				self.handleError(error);
 | 
						|
			});
 | 
						|
		}).on('error', function(error) {
 | 
						|
			self.handleError(error);
 | 
						|
		});
 | 
						|
 | 
						|
		req.setHeader("Connection", "Close");
 | 
						|
 | 
						|
		// Node 0.4 and later won't accept empty data. Make sure it's needed.
 | 
						|
		if (data) {
 | 
						|
			req.write(data);
 | 
						|
		}
 | 
						|
 | 
						|
		req.end();
 | 
						|
	};
 | 
						|
 | 
						|
	this.handleError = function(error) {
 | 
						|
		this.status = 503;
 | 
						|
		this.statusText = error;
 | 
						|
		this.responseText = error.stack;
 | 
						|
		setState(this.DONE);
 | 
						|
	};
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Aborts a request.
 | 
						|
	 */
 | 
						|
	this.abort = function() {
 | 
						|
		headers = defaultHeaders;
 | 
						|
		this.readyState = this.UNSENT;
 | 
						|
		this.responseText = "";
 | 
						|
		this.responseXML = "";
 | 
						|
	};
 | 
						|
	
 | 
						|
	/**
 | 
						|
	 * Changes readyState and calls onreadystatechange.
 | 
						|
	 *
 | 
						|
	 * @param int state New state
 | 
						|
	 */
 | 
						|
	var setState = function(state) {
 | 
						|
		self.readyState = state;
 | 
						|
		self.onreadystatechange();
 | 
						|
	}
 | 
						|
};
 |