Skip to content

Commit

Permalink
Initial testing infrastructure
Browse files Browse the repository at this point in the history
Slight rearrangement of directories to support tests. Add some initial
tests. Tests can be run using scripts/test.sh. Note that an Uncaught
TypeError results when the tests are run that can be safely ignored for
now. A fix is in progress.
  • Loading branch information
Robert Gardner committed Dec 6, 2013
1 parent 5088e1c commit 0cd9ca0
Show file tree
Hide file tree
Showing 17 changed files with 28,533 additions and 25 deletions.
12 changes: 7 additions & 5 deletions app/js/app.js
Original file line number Diff line number Diff line change
Expand Up @@ -142,8 +142,11 @@ function buildWeRelateUrl($scope) {
if ($scope.deathPlace) {
query += '&dp=' + encodeURIComponent($scope.deathPlace);
}
var search = 'http://www.werelate.org/wiki/Special:Search?ns=Person' + query;
search += '&k=' + encodeURIComponent($scope.keywords) + '&rows=20&ecp=p';
if ($scope.keywords) {
query += '&k=' + encodeURIComponent($scope.keywords);
}
var search = 'http://www.werelate.org/wiki/Special:Search?ns=Person'
+ query + '&rows=20&ecp=p';
return search;
}

Expand Down Expand Up @@ -189,8 +192,7 @@ function buildGeniUrl($scope) {
* @return {string} The resulting query URL.
*/
function buildPeopleSearchUrl($scope) {
return 'https://www.google.com/search?' +
'tbs=ppl:1&e=ForceExperiment&expid=52793&q=' +
return 'https://www.google.com/search?q=' +
buildGoogleQuery($scope) + '+~genealogy';
}

Expand Down Expand Up @@ -271,7 +273,7 @@ function buildGoogleQuery($scope, opt_excludeDate) {
}
}
if ($scope.keywords) {
q += encodeURIComponent(' ' + $scope.keywords);
q += '+' + encodeURIComponent($scope.keywords);
}
return q;
}
Expand Down
20 changes: 0 additions & 20 deletions app/js/util.js
Original file line number Diff line number Diff line change
Expand Up @@ -121,23 +121,3 @@ Util.format = function(var_args) {
return args[i];
});
};

Util.sortByDate = function(a, b) {
if (a.updatedDateFull < b.updatedDateFull) {
return 1;
}
if (a.updatedDateFull > b.updatedDateFull) {
return -1;
}
return 0;
}

Util.sortByTitle = function(a, b) {
if (a.title < b.title) {
return 1;
}
if (a.title > b.title) {
return -1;
}
return 0;
}
File renamed without changes.
26 changes: 26 additions & 0 deletions config/karma-e2e.conf.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
module.exports = function (config) {
config.set({
basePath: '../',

files: [
'test/e2e/**/*.js'
],

frameworks: ['ng-scenario'],

autoWatch: false,

browsers: ['Chrome'],

singleRun: true,

proxies: {
'/': 'http://localhost:8000/'
},

junitReporter: {
outputFile: 'test_out/e2e.xml',
suite: 'e2e'
}
});
};
24 changes: 24 additions & 0 deletions config/karma.conf.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
module.exports = function (config) {
config.set({
basePath: '../',

files: [
'app/lib/angular-*.js',
'test/lib/angular/angular-mocks.js',
'app/js/util.js',
'app/js/app.js',
'test/unit/**/*.js'
],

frameworks: ['jasmine'],

autoWatch: true,

browsers: ['Chrome'],

junitReporter: {
outputFile: 'test_out/unit.xml',
suite: 'unit'
}
});
};
9 changes: 9 additions & 0 deletions scripts/e2e-test.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
#!/bin/bash

BASE_DIR=`dirname $0`

echo ""
echo "Starting Karma Server (http://karma-runner.github.io)"
echo "-------------------------------------------------------------------"

karma start $BASE_DIR/../config/karma-e2e.conf.js $*
9 changes: 9 additions & 0 deletions scripts/test.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
#!/bin/bash

BASE_DIR=`dirname $0`

echo ""
echo "Starting Karma Server (http://karma-runner.github.io)"
echo "-------------------------------------------------------------------"

karma start $BASE_DIR/../config/karma.conf.js $*
244 changes: 244 additions & 0 deletions scripts/web-server.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,244 @@
#!/usr/bin/env node

var util = require('util'),
http = require('http'),
fs = require('fs'),
url = require('url'),
events = require('events');

var DEFAULT_PORT = 8000;

function main(argv) {
new HttpServer({
'GET': createServlet(StaticServlet),
'HEAD': createServlet(StaticServlet)
}).start(Number(argv[2]) || DEFAULT_PORT);
}

function escapeHtml(value) {
return value.toString().
replace('<', '&lt;').
replace('>', '&gt;').
replace('"', '&quot;');
}

function createServlet(Class) {
var servlet = new Class();
return servlet.handleRequest.bind(servlet);
}

/**
* An Http server implementation that uses a map of methods to decide
* action routing.
*
* @param {Object} Map of method => Handler function
*/
function HttpServer(handlers) {
this.handlers = handlers;
this.server = http.createServer(this.handleRequest_.bind(this));
}

HttpServer.prototype.start = function(port) {
this.port = port;
this.server.listen(port);
util.puts('Http Server running at http://localhost:' + port + '/');
};

HttpServer.prototype.parseUrl_ = function(urlString) {
var parsed = url.parse(urlString);
parsed.pathname = url.resolve('/', parsed.pathname);
return url.parse(url.format(parsed), true);
};

HttpServer.prototype.handleRequest_ = function(req, res) {
var logEntry = req.method + ' ' + req.url;
if (req.headers['user-agent']) {
logEntry += ' ' + req.headers['user-agent'];
}
util.puts(logEntry);
req.url = this.parseUrl_(req.url);
var handler = this.handlers[req.method];
if (!handler) {
res.writeHead(501);
res.end();
} else {
handler.call(this, req, res);
}
};

/**
* Handles static content.
*/
function StaticServlet() {}

StaticServlet.MimeMap = {
'txt': 'text/plain',
'html': 'text/html',
'css': 'text/css',
'xml': 'application/xml',
'json': 'application/json',
'js': 'application/javascript',
'jpg': 'image/jpeg',
'jpeg': 'image/jpeg',
'gif': 'image/gif',
'png': 'image/png',
  'svg': 'image/svg+xml'
};

StaticServlet.prototype.handleRequest = function(req, res) {
var self = this;
var path = ('./' + req.url.pathname).replace('//','/').replace(/%(..)/g, function(match, hex){
return String.fromCharCode(parseInt(hex, 16));
});
var parts = path.split('/');
if (parts[parts.length-1].charAt(0) === '.')
return self.sendForbidden_(req, res, path);
fs.stat(path, function(err, stat) {
if (err)
return self.sendMissing_(req, res, path);
if (stat.isDirectory())
return self.sendDirectory_(req, res, path);
return self.sendFile_(req, res, path);
});
}

StaticServlet.prototype.sendError_ = function(req, res, error) {
res.writeHead(500, {
'Content-Type': 'text/html'
});
res.write('<!doctype html>\n');
res.write('<title>Internal Server Error</title>\n');
res.write('<h1>Internal Server Error</h1>');
res.write('<pre>' + escapeHtml(util.inspect(error)) + '</pre>');
util.puts('500 Internal Server Error');
util.puts(util.inspect(error));
};

StaticServlet.prototype.sendMissing_ = function(req, res, path) {
path = path.substring(1);
res.writeHead(404, {
'Content-Type': 'text/html'
});
res.write('<!doctype html>\n');
res.write('<title>404 Not Found</title>\n');
res.write('<h1>Not Found</h1>');
res.write(
'<p>The requested URL ' +
escapeHtml(path) +
' was not found on this server.</p>'
);
res.end();
util.puts('404 Not Found: ' + path);
};

StaticServlet.prototype.sendForbidden_ = function(req, res, path) {
path = path.substring(1);
res.writeHead(403, {
'Content-Type': 'text/html'
});
res.write('<!doctype html>\n');
res.write('<title>403 Forbidden</title>\n');
res.write('<h1>Forbidden</h1>');
res.write(
'<p>You do not have permission to access ' +
escapeHtml(path) + ' on this server.</p>'
);
res.end();
util.puts('403 Forbidden: ' + path);
};

StaticServlet.prototype.sendRedirect_ = function(req, res, redirectUrl) {
res.writeHead(301, {
'Content-Type': 'text/html',
'Location': redirectUrl
});
res.write('<!doctype html>\n');
res.write('<title>301 Moved Permanently</title>\n');
res.write('<h1>Moved Permanently</h1>');
res.write(
'<p>The document has moved <a href="' +
redirectUrl +
'">here</a>.</p>'
);
res.end();
util.puts('301 Moved Permanently: ' + redirectUrl);
};

StaticServlet.prototype.sendFile_ = function(req, res, path) {
var self = this;
var file = fs.createReadStream(path);
res.writeHead(200, {
'Content-Type': StaticServlet.
MimeMap[path.split('.').pop()] || 'text/plain'
});
if (req.method === 'HEAD') {
res.end();
} else {
file.on('data', res.write.bind(res));
file.on('close', function() {
res.end();
});
file.on('error', function(error) {
self.sendError_(req, res, error);
});
}
};

StaticServlet.prototype.sendDirectory_ = function(req, res, path) {
var self = this;
if (path.match(/[^\/]$/)) {
req.url.pathname += '/';
var redirectUrl = url.format(url.parse(url.format(req.url)));
return self.sendRedirect_(req, res, redirectUrl);
}
fs.readdir(path, function(err, files) {
if (err)
return self.sendError_(req, res, error);

if (!files.length)
return self.writeDirectoryIndex_(req, res, path, []);

var remaining = files.length;
files.forEach(function(fileName, index) {
fs.stat(path + '/' + fileName, function(err, stat) {
if (err)
return self.sendError_(req, res, err);
if (stat.isDirectory()) {
files[index] = fileName + '/';
}
if (!(--remaining))
return self.writeDirectoryIndex_(req, res, path, files);
});
});
});
};

StaticServlet.prototype.writeDirectoryIndex_ = function(req, res, path, files) {
path = path.substring(1);
res.writeHead(200, {
'Content-Type': 'text/html'
});
if (req.method === 'HEAD') {
res.end();
return;
}
res.write('<!doctype html>\n');
res.write('<title>' + escapeHtml(path) + '</title>\n');
res.write('<style>\n');
res.write(' ol { list-style-type: none; font-size: 1.2em; }\n');
res.write('</style>\n');
res.write('<h1>Directory: ' + escapeHtml(path) + '</h1>');
res.write('<ol>');
files.forEach(function(fileName) {
if (fileName.charAt(0) !== '.') {
res.write('<li><a href="' +
escapeHtml(fileName) + '">' +
escapeHtml(fileName) + '</a></li>');
}
});
res.write('</ol>');
res.end();
};

// Must be last,
main(process.argv);
11 changes: 11 additions & 0 deletions test/e2e/runner.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
<!doctype html>
<html lang="en">
<head>
<title>End2end Test Runner</title>
<meta charset="utf-8">
<script src="../lib/angular/angular-scenario.js" ng-autotest></script>
<script src="scenarios.js"></script>
</head>
<body>
</body>
</html>
Loading

0 comments on commit 0cd9ca0

Please sign in to comment.