gecko-dev/services/sync/tests/unit/test_httpd_sync_server.js

286 lines
9.2 KiB
JavaScript

/* Any copyright is dedicated to the Public Domain.
http://creativecommons.org/publicdomain/zero/1.0/ */
Cu.import("resource://services-sync/util.js");
function run_test() {
Log.repository.getLogger("Sync.Test.Server").level = Log.Level.Trace;
initTestLogging();
run_next_test();
}
add_test(function test_creation() {
// Explicit callback for this one.
let server = new SyncServer({
__proto__: SyncServerCallback,
});
do_check_true(!!server); // Just so we have a check.
server.start(null, function () {
_("Started on " + server.port);
server.stop(run_next_test);
});
});
add_test(function test_url_parsing() {
let server = new SyncServer();
// Check that we can parse a WBO URI.
let parts = server.pathRE.exec("/1.1/johnsmith/storage/crypto/keys");
let [all, version, username, first, rest] = parts;
do_check_eq(all, "/1.1/johnsmith/storage/crypto/keys");
do_check_eq(version, "1.1");
do_check_eq(username, "johnsmith");
do_check_eq(first, "storage");
do_check_eq(rest, "crypto/keys");
do_check_eq(null, server.pathRE.exec("/nothing/else"));
// Check that we can parse a collection URI.
parts = server.pathRE.exec("/1.1/johnsmith/storage/crypto");
[all, version, username, first, rest] = parts;
do_check_eq(all, "/1.1/johnsmith/storage/crypto");
do_check_eq(version, "1.1");
do_check_eq(username, "johnsmith");
do_check_eq(first, "storage");
do_check_eq(rest, "crypto");
// We don't allow trailing slash on storage URI.
parts = server.pathRE.exec("/1.1/johnsmith/storage/");
do_check_eq(parts, undefined);
// storage alone is a valid request.
parts = server.pathRE.exec("/1.1/johnsmith/storage");
[all, version, username, first, rest] = parts;
do_check_eq(all, "/1.1/johnsmith/storage");
do_check_eq(version, "1.1");
do_check_eq(username, "johnsmith");
do_check_eq(first, "storage");
do_check_eq(rest, undefined);
parts = server.storageRE.exec("storage");
let storage, collection, id;
[all, storage, collection, id] = parts;
do_check_eq(all, "storage");
do_check_eq(collection, undefined);
run_next_test();
});
Cu.import("resource://services-common/rest.js");
function localRequest(server, path) {
_("localRequest: " + path);
let url = server.baseURI.substr(0, server.baseURI.length - 1) + path;
_("url: " + url);
return new RESTRequest(url);
}
add_test(function test_basic_http() {
let server = new SyncServer();
server.registerUser("john", "password");
do_check_true(server.userExists("john"));
server.start(null, function () {
_("Started on " + server.port);
Utils.nextTick(function () {
let req = localRequest(server, "/1.1/john/storage/crypto/keys");
_("req is " + req);
req.get(function (err) {
do_check_eq(null, err);
Utils.nextTick(function () {
server.stop(run_next_test);
});
});
});
});
});
add_test(function test_info_collections() {
let server = new SyncServer({
__proto__: SyncServerCallback
});
function responseHasCorrectHeaders(r) {
do_check_eq(r.status, 200);
do_check_eq(r.headers["content-type"], "application/json");
do_check_true("x-weave-timestamp" in r.headers);
}
server.registerUser("john", "password");
server.start(null, function () {
Utils.nextTick(function () {
let req = localRequest(server, "/1.1/john/info/collections");
req.get(function (err) {
// Initial info/collections fetch is empty.
do_check_eq(null, err);
responseHasCorrectHeaders(this.response);
do_check_eq(this.response.body, "{}");
Utils.nextTick(function () {
// When we PUT something to crypto/keys, "crypto" appears in the response.
function cb(err) {
do_check_eq(null, err);
responseHasCorrectHeaders(this.response);
let putResponseBody = this.response.body;
_("PUT response body: " + JSON.stringify(putResponseBody));
req = localRequest(server, "/1.1/john/info/collections");
req.get(function (err) {
do_check_eq(null, err);
responseHasCorrectHeaders(this.response);
let expectedColl = server.getCollection("john", "crypto");
do_check_true(!!expectedColl);
let modified = expectedColl.timestamp;
do_check_true(modified > 0);
do_check_eq(putResponseBody, modified);
do_check_eq(JSON.parse(this.response.body).crypto, modified);
Utils.nextTick(function () {
server.stop(run_next_test);
});
});
}
let payload = JSON.stringify({foo: "bar"});
localRequest(server, "/1.1/john/storage/crypto/keys").put(payload, cb);
});
});
});
});
});
add_test(function test_storage_request() {
let keysURL = "/1.1/john/storage/crypto/keys?foo=bar";
let foosURL = "/1.1/john/storage/crypto/foos";
let storageURL = "/1.1/john/storage";
let server = new SyncServer();
let creation = server.timestamp();
server.registerUser("john", "password");
server.createContents("john", {
crypto: {foos: {foo: "bar"}}
});
let coll = server.user("john").collection("crypto");
do_check_true(!!coll);
_("We're tracking timestamps.");
do_check_true(coll.timestamp >= creation);
function retrieveWBONotExists(next) {
let req = localRequest(server, keysURL);
req.get(function (err) {
_("Body is " + this.response.body);
_("Modified is " + this.response.newModified);
do_check_eq(null, err);
do_check_eq(this.response.status, 404);
do_check_eq(this.response.body, "Not found");
Utils.nextTick(next);
});
}
function retrieveWBOExists(next) {
let req = localRequest(server, foosURL);
req.get(function (err) {
_("Body is " + this.response.body);
_("Modified is " + this.response.newModified);
let parsedBody = JSON.parse(this.response.body);
do_check_eq(parsedBody.id, "foos");
do_check_eq(parsedBody.modified, coll.wbo("foos").modified);
do_check_eq(JSON.parse(parsedBody.payload).foo, "bar");
Utils.nextTick(next);
});
}
function deleteWBONotExists(next) {
let req = localRequest(server, keysURL);
server.callback.onItemDeleted = function (username, collection, wboID) {
do_throw("onItemDeleted should not have been called.");
};
req.delete(function (err) {
_("Body is " + this.response.body);
_("Modified is " + this.response.newModified);
do_check_eq(this.response.status, 200);
delete server.callback.onItemDeleted;
Utils.nextTick(next);
});
}
function deleteWBOExists(next) {
let req = localRequest(server, foosURL);
server.callback.onItemDeleted = function (username, collection, wboID) {
_("onItemDeleted called for " + collection + "/" + wboID);
delete server.callback.onItemDeleted;
do_check_eq(username, "john");
do_check_eq(collection, "crypto");
do_check_eq(wboID, "foos");
Utils.nextTick(next);
};
req.delete(function (err) {
_("Body is " + this.response.body);
_("Modified is " + this.response.newModified);
do_check_eq(this.response.status, 200);
});
}
function deleteStorage(next) {
_("Testing DELETE on /storage.");
let now = server.timestamp();
_("Timestamp: " + now);
let req = localRequest(server, storageURL);
req.delete(function (err) {
_("Body is " + this.response.body);
_("Modified is " + this.response.newModified);
let parsedBody = JSON.parse(this.response.body);
do_check_true(parsedBody >= now);
do_check_empty(server.users["john"].collections);
Utils.nextTick(next);
});
}
function getStorageFails(next) {
_("Testing that GET on /storage fails.");
let req = localRequest(server, storageURL);
req.get(function (err) {
do_check_eq(this.response.status, 405);
do_check_eq(this.response.headers["allow"], "DELETE");
Utils.nextTick(next);
});
}
function getMissingCollectionWBO(next) {
_("Testing that fetching a WBO from an on-existent collection 404s.");
let req = localRequest(server, storageURL + "/foobar/baz");
req.get(function (err) {
do_check_eq(this.response.status, 404);
Utils.nextTick(next);
});
}
server.start(null,
Async.chain(
retrieveWBONotExists,
retrieveWBOExists,
deleteWBOExists,
deleteWBONotExists,
getStorageFails,
getMissingCollectionWBO,
deleteStorage,
server.stop.bind(server),
run_next_test
));
});
add_test(function test_x_weave_records() {
let server = new SyncServer();
server.registerUser("john", "password");
server.createContents("john", {
crypto: {foos: {foo: "bar"},
bars: {foo: "baz"}}
});
server.start(null, function () {
let wbo = localRequest(server, "/1.1/john/storage/crypto/foos");
wbo.get(function (err) {
// WBO fetches don't have one.
do_check_false("x-weave-records" in this.response.headers);
let col = localRequest(server, "/1.1/john/storage/crypto");
col.get(function (err) {
// Collection fetches do.
do_check_eq(this.response.headers["x-weave-records"], "2");
server.stop(run_next_test);
});
});
});
});