/// import http = require("http"); import fs = require("fs"); import path = require("path"); import url = require("url"); import child_process = require("child_process"); /// Command line processing /// if (process.argv[2] == '--help') { console.log('Runs a node server on port 8888 by default, looking for tests folder in the current directory\n'); console.log('Syntax: node nodeServer.js [port] [typescriptEnlistmentDirectory] [tests] [--browser] [--verbose]\n'); console.log('Examples: \n\tnode nodeServer.js 8888 .'); console.log('\tnode nodeServer.js 3000 D:/src/typescript/public --verbose IE'); } function switchToForwardSlashes(path: string) { return path.replace(/\\/g, "/").replace(/\/\//g, '/'); } var defaultPort = 8888; var port = process.argv[2] || defaultPort; var rootDir = switchToForwardSlashes(__dirname + '/../'); var browser: string; if (process.argv[3]) { browser = process.argv[3]; if (browser !== 'chrome' && browser !== 'IE') { console.log('Invalid command line arguments. Got ' + browser + ' but expected chrome, IE or nothing.'); } } var grep = process.argv[4]; var verbose = false; if (process.argv[5] == '--verbose') { verbose = true; } else if (process.argv[5] && process.argv[5] !== '--verbose') { console.log('Invalid command line arguments. Got ' + process.argv[5] + ' but expected --verbose or nothing.'); } /// Utils /// function log(msg: string) { if (verbose) { console.log(msg); } } // Copied from the compiler sources function dir(path: string, spec?: string, options?: any) { options = options || <{ recursive?: boolean; }>{}; function filesInFolder(folder: string): string[] { var folder = switchToForwardSlashes(folder); var paths: string[] = []; // Everything after the current directory is relative var baseDirectoryLength = process.cwd().length + 1; try { var files = fs.readdirSync(folder); for (var i = 0; i < files.length; i++) { var stat = fs.statSync(folder + "/" + files[i]); if (options.recursive && stat.isDirectory()) { paths = paths.concat(filesInFolder(folder + "/" + files[i])); } else if (stat.isFile() && (!spec || files[i].match(spec))) { var relativePath = folder.substring(baseDirectoryLength); paths.push(relativePath + "/" + files[i]); } } } catch (err) { // Skip folders that are inaccessible } return paths; } return filesInFolder(path); } // fs.rmdirSync won't delete directories with files in it function deleteFolderRecursive(path: string) { if (fs.existsSync(path)) { fs.readdirSync(path).forEach(function (file, index) { var curPath = path + "/" + file; if (fs.statSync(curPath).isDirectory()) { // recurse deleteFolderRecursive(curPath); } else { // delete file fs.unlinkSync(curPath); } }); fs.rmdirSync(path); } }; function writeFile(path: string, data: any, opts: { recursive: boolean }) { try { fs.writeFileSync(path, data); } catch (e) { // assume file was written to a directory that exists, if not, start recursively creating them as necessary var parts = switchToForwardSlashes(path).split('/'); for (var i = 0; i < parts.length; i++) { var subDir = parts.slice(0, i).join('/'); if (!fs.existsSync(subDir)) { fs.mkdir(subDir); } } fs.writeFileSync(path, data); } } /// Request Handling /// function handleResolutionRequest(filePath: string, res: http.ServerResponse) { var resolvedPath = path.resolve(filePath, ''); resolvedPath = resolvedPath.substring(resolvedPath.indexOf('tests')); resolvedPath = switchToForwardSlashes(resolvedPath); send('success', res, resolvedPath); return; } function send(result: "fail", res: http.ServerResponse, contents: string, contentType?: string): void; function send(result: "success", res: http.ServerResponse, contents: string, contentType?: string): void; function send(result: "unknown", res: http.ServerResponse, contents: string, contentType?: string): void; function send(result: string, res: http.ServerResponse, contents: string, contentType?: string): void function send(result: string, res: http.ServerResponse, contents: string, contentType = "binary"): void { var responseCode = result === "success" ? 200 : result === "fail" ? 500 : result === 'unknown' ? 404 : parseInt(result); res.writeHead(responseCode, { "Content-Type": contentType }); res.end(contents); return; } // Reads the data from a post request and passes it to the given callback function processPost(req: http.ServerRequest, res: http.ServerResponse, callback: (data: string) => any): void { var queryData = ""; if (typeof callback !== 'function') return null; if (req.method == 'POST') { req.on('data', function (data: string) { queryData += data; if (queryData.length > 1e8) { queryData = ""; send("413", res, null); console.log("ERROR: destroying connection"); req.connection.destroy(); } }); req.on('end', function () { //res.post = url.parse(req.url).query; callback(queryData); }); } else { send("405", res, null); } } enum RequestType { GetFile, GetDir, ResolveFile, WriteFile, DeleteFile, WriteDir, DeleteDir, AppendFile, Unknown } function getRequestOperation(req: http.ServerRequest, filename: string) { if (req.method === 'GET' && req.url.indexOf('?') === -1) { if (req.url.indexOf('.') !== -1) return RequestType.GetFile; else return RequestType.GetDir; } else { var queryData: any = url.parse(req.url, true).query; if (req.method === 'GET' && queryData.resolve !== undefined) return RequestType.ResolveFile // mocha uses ?grep= query string as equivalent to the --grep command line option used to filter tests if (req.method === 'GET' && queryData.grep !== undefined) return RequestType.GetFile if (req.method === 'POST' && queryData.action) { var path = req.url.substr(0, req.url.lastIndexOf('?')); var isFile = path.substring(path.lastIndexOf('/')).indexOf('.') !== -1; switch (queryData.action.toUpperCase()) { case 'WRITE': return isFile ? RequestType.WriteFile : RequestType.WriteDir; case 'DELETE': return isFile ? RequestType.DeleteFile : RequestType.DeleteDir; case 'APPEND': return isFile ? RequestType.AppendFile : RequestType.Unknown; } } return RequestType.Unknown } return RequestType.Unknown } function handleRequestOperation(req: http.ServerRequest, res: http.ServerResponse, operation: RequestType, reqPath: string) { switch (operation) { case RequestType.GetDir: var filesInFolder = dir(reqPath, "", { recursive: true }); send('success', res, filesInFolder.join(',')); break; case RequestType.GetFile: fs.readFile(reqPath, function (err, file) { var ext = reqPath.substr(reqPath.lastIndexOf('.')); var contentType = 'binary'; if (ext === '.js') contentType = 'text/javascript' else if (ext === '.css') contentType = 'text/javascript' else if (ext === '.html') contentType = 'text/html' err ? send('fail', res, err.message, contentType) : send('success', res, (file), contentType); }); break; case RequestType.ResolveFile: var resolveRequest = req.url.match(/(.*)\?resolve/); handleResolutionRequest(resolveRequest[1], res); break; case RequestType.WriteFile: processPost(req, res, (data) => { writeFile(reqPath, data, { recursive: true }); }); send('success', res, null); break; case RequestType.WriteDir: fs.mkdirSync(reqPath); send('success', res, null); break; case RequestType.DeleteFile: if (fs.existsSync(reqPath)) { fs.unlinkSync(reqPath); } send('success', res, null); break; case RequestType.DeleteDir: if (fs.existsSync(reqPath)) { fs.rmdirSync(reqPath); } send('success', res, null); break; case RequestType.AppendFile: processPost(req, res, (data) => { fs.appendFileSync(reqPath, data); }); send('success', res, null); break; case RequestType.Unknown: default: send('unknown', res, null); break; } } console.log("Static file server running at\n => http://localhost:" + port + "/\nCTRL + C to shutdown"); http.createServer(function (req: http.ServerRequest, res: http.ServerResponse) { log(req.method + ' ' + req.url); var uri = url.parse(req.url).pathname var reqPath = path.join(process.cwd(), uri); var operation = getRequestOperation(req, reqPath); handleRequestOperation(req, res, operation, reqPath); }).listen(8888); var browserPath: string; if ((browser && browser === 'chrome')) { var defaultChromePath = "C:/Program Files (x86)/Google/Chrome/Application/chrome.exe"; if (fs.existsSync(defaultChromePath)) { browserPath = defaultChromePath; } else { browserPath = browser; } } else { var defaultIEPath = 'C:/Program Files/Internet Explorer/iexplore.exe'; if (fs.existsSync(defaultIEPath)) { browserPath = defaultIEPath; } else { browserPath = browser; } } console.log('Using browser: ' + browserPath); var queryString = grep ? "?grep=" + grep : ''; child_process.spawn(browserPath, ['http://localhost:' + port + '/tests/webTestResults.html' + queryString], { stdio: 'inherit' });