| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043 | /* globals describe, it, before, after *//* eslint no-loop-func:0 guard-for-in:0 */var networking = require("../lib/networking");var helpers = require("../lib/helpers");var logging = require("../lib/logging");var cleaner = require("../lib/cleaner");var request = require("request");var config = require("../config");var server = require("../lib/server");var assert = require("assert");var skins = require("../lib/skins");var cache = require("../lib/cache");var crc = require("crc").crc32;var fs = require("fs");// we don't want tests to fail because of slow internetconfig.server.http_timeout *= 3;// no spamif (process.env.VERBOSE_TEST !== "true") {  logging.log = logging.debug = logging.warn = logging.error = function() {};}var uuids = fs.readFileSync("test/uuids.txt").toString().split(/\r?\n/);var names = fs.readFileSync("test/usernames.txt").toString().split(/\r?\n/);// Get a random UUID + name in order to prevent rate limitingvar uuid = uuids[Math.round(Math.random() * (uuids.length - 1))];var name = names[Math.round(Math.random() * (names.length - 1))];// Let's hope these will never be assignedvar steve_ids = [  "fffffff0" + "fffffff0" + "fffffff0" + "fffffff0",  "fffffff0" + "fffffff0" + "fffffff1" + "fffffff1",  "fffffff0" + "fffffff1" + "fffffff0" + "fffffff1",  "fffffff0" + "fffffff1" + "fffffff1" + "fffffff0",  "fffffff1" + "fffffff0" + "fffffff0" + "fffffff1",  "fffffff1" + "fffffff0" + "fffffff1" + "fffffff0",  "fffffff1" + "fffffff1" + "fffffff0" + "fffffff0",  "fffffff1" + "fffffff1" + "fffffff1" + "fffffff1",];// Let's hope these will never be assignedvar alex_ids = [  "fffffff0" + "fffffff0" + "fffffff0" + "fffffff1",  "fffffff0" + "fffffff0" + "fffffff1" + "fffffff0",  "fffffff0" + "fffffff1" + "fffffff0" + "fffffff0",  "fffffff0" + "fffffff1" + "fffffff1" + "fffffff1",  "fffffff1" + "fffffff0" + "fffffff0" + "fffffff0",  "fffffff1" + "fffffff0" + "fffffff1" + "fffffff1",  "fffffff1" + "fffffff1" + "fffffff0" + "fffffff1",  "fffffff1" + "fffffff1" + "fffffff1" + "fffffff0",];var rid = "TestReqID: ";function getRandomInt(min, max) {  return Math.floor(Math.random() * (max - min + 1)) + min;}var ids = [  uuid.toLowerCase(),  name.toLowerCase(),  name.toUpperCase(),  uuid.toUpperCase(),];describe("Crafatar", function() {  // we might have to make 2 HTTP requests  this.timeout(config.server.http_timeout * 2 + 50);  before(function() {    cache.get_redis().flushall();    // cause I don't know how big hard drives are these days    config.cleaner.disk_limit = Infinity;    config.cleaner.redis_limit = Infinity;    cleaner.run();  });  describe("UUID/username", function() {    it("non-hex uuid is invalid", function(done) {      assert.strictEqual(helpers.id_valid("g098cb60fa8e427cb299793cbd302c9a"), false);      done();    });    it("empty id is invalid", function(done) {      assert.strictEqual(helpers.id_valid(""), false);      done();    });    it("non-alphanumeric username is invalid", function(done) {      assert.strictEqual(helpers.id_valid("usernäme"), false);      done();    });    it("dashed username is invalid", function(done) {      assert.strictEqual(helpers.id_valid("user-name"), false);      done();    });    it(">16 length username is invalid", function(done) {      assert.strictEqual(helpers.id_valid("ThisNameIsTooLong"), false);      done();    });    it("lowercase uuid is valid", function(done) {      assert.strictEqual(helpers.id_valid("0098cb60fa8e427cb299793cbd302c9a"), true);      done();    });    it("uppercase uuid is valid", function(done) {      assert.strictEqual(helpers.id_valid("1DCEF164FF0A47F2B9A691385C774EE7"), true);      done();    });    it("dashed uuid is valid", function(done) {      assert.strictEqual(helpers.id_valid("0098cb60-fa8e-427c-b299-793cbd302c9a"), true);      done();    });    it("16 chars, underscored, capital, numbered username is valid", function(done) {      assert.strictEqual(helpers.id_valid("__niceUs3rname__"), true);      done();    });    it("1 char username is valid", function(done) {      assert.strictEqual(helpers.id_valid("a"), true);      done();    });    it("should not exist (uuid)", function(done) {      var number = getRandomInt(0, 9).toString();      networking.get_profile(rid, Array(33).join(number), function(err, profile) {        assert.ifError(err);        assert.strictEqual(profile, null);        done();      });    });    it("should not exist (username)", function(done) {      networking.get_username_url(rid, "Steve", 0, function(err, profile) {        assert.ifError(err);        done();      });    });  });  describe("Avatar", function() {    it("uuid's account should exist, but skin should not", function(done) {      // profile "Alex" - hoping it'll never have a skin      networking.get_profile(rid, "ec561538f3fd461daff5086b22154bce", function(err, profile) {        assert.ifError(err);        assert.notStrictEqual(profile, null);        networking.get_uuid_url(profile, 1, function(url) {          assert.strictEqual(url, null);          done();        });      });    });    it("Username should default to Steve", function(done) {      assert.strictEqual(skins.default_skin("TestUser"), "steve");      done();    });    for (var a in alex_ids) {      var alexid = alex_ids[a];      (function(alex_id) {        it("UUID " + alex_id + " should default to Alex", function(done) {          assert.strictEqual(skins.default_skin(alex_id), "alex");          done();        });      }(alexid));    }    for (var s in steve_ids) {      var steveid = steve_ids[s];      (function(steve_id) {        it("UUID " + steve_id + " should default to Steve", function(done) {          assert.strictEqual(skins.default_skin(steve_id), "steve");          done();        });      }(steveid));    }  });  describe("Errors", function() {    it("should time out on uuid info download", function(done) {      var original_timeout = config.server.http_timeout;      config.server.http_timeout = 1;      networking.get_profile(rid, "069a79f444e94726a5befca90e38aaf5", function(err, profile) {        assert.strictEqual(err.code, "ETIMEDOUT");        config.server.http_timeout = original_timeout;        done();      });    });    it("should time out on username info download", function(done) {      var original_timeout = config.server.http_timeout;      config.server.http_timeout = 1;      networking.get_username_url(rid, "jomo", 0, function(err, url) {        assert.strictEqual(err.code, "ETIMEDOUT");        config.server.http_timeout = original_timeout;        done();      });    });    it("should time out on skin download", function(done) {      var original_timeout = config.http_timeout;      config.server.http_timeout = 1;      networking.get_from(rid, "http://textures.minecraft.net/texture/477be35554684c28bdeee4cf11c591d3c88afb77e0b98da893fd7bc318c65184", function(body, res, error) {        assert.strictEqual(error.code, "ETIMEDOUT");        config.server.http_timeout = original_timeout;        done();      });    });    it("should not find the skin", function(done) {      assert.doesNotThrow(function() {        networking.get_from(rid, "http://textures.minecraft.net/texture/this-does-not-exist", function(img, response, err) {          assert.strictEqual(err, null); // no error here, but it shouldn't throw exceptions          done();        });      });    });    it("should ignore file updates on invalid files", function(done) {      assert.doesNotThrow(function() {        cache.update_timestamp(rid, "0123456789abcdef0123456789abcdef", "invalid-file.png", false, function(err) {          assert.ifError(err);          done();        });      });    });    it("should not find the file", function(done) {      skins.open_skin(rid, "non/existent/path", function(err, img) {        assert(err);        done();      });    });  });  describe("Server", function() {    // throws Exception when default headers are not in res.headers    function assert_headers(res) {      assert(res.headers["content-type"]);      assert("" + res.headers["response-time"]);      assert(res.headers["x-request-id"]);      assert.equal(res.headers["access-control-allow-origin"], "*");      assert.equal(res.headers["cache-control"], "max-age=" + config.caching.browser + ", public");    }    // throws Exception when +url+ is requested with +etag+    // and it does not return 304 without a body    function assert_cache(url, etag, callback) {      request.get(url, {        headers: {          "If-None-Match": etag        }      }, function(error, res, body) {        assert.ifError(error);        assert.ifError(body);        assert.equal(res.statusCode, 304);        assert(res.headers.etag);        assert_headers(res);        callback();      });    }    before(function(done) {      server.boot(function() {        done();      });    });    it("should return 405 Method Not Allowed for POST", function(done) {      request.post("http://localhost:3000", function(error, res, body) {        assert.ifError(error);        assert.strictEqual(res.statusCode, 405);        done();      });    });    it("should return correct HTTP response for home page", function(done) {      var url = "http://localhost:3000";      request.get(url, function(error, res, body) {        assert.ifError(error);        assert.strictEqual(res.statusCode, 200);        assert_headers(res);        assert(res.headers.etag);        assert.strictEqual(res.headers["content-type"], "text/html; charset=utf-8");        assert.strictEqual(res.headers.etag, '"' + crc(body) + '"');        assert(body);        assert_cache(url, res.headers.etag, function() {          done();        });      });    });    it("should return correct HTTP response for assets", function(done) {      var url = "http://localhost:3000/stylesheets/style.css";      request.get(url, function(error, res, body) {        assert.ifError(error);        assert.strictEqual(res.statusCode, 200);        assert_headers(res);        assert(res.headers.etag);        assert.strictEqual(res.headers["content-type"], "text/css");        assert.strictEqual(res.headers.etag, '"' + crc(body) + '"');        assert(body);        assert_cache(url, res.headers.etag, function() {          done();        });      });    });    it("should return correct HTTP response for URL encoded URLs", function(done) {      var url = "http://localhost:3000/%61%76%61%74%61%72%73/%6a%6f%6d%6f"; // avatars/jomo      request.get(url, function(error, res, body) {        assert.ifError(error);        assert.strictEqual(res.statusCode, 200);        assert_headers(res);        assert(res.headers.etag);        assert.strictEqual(res.headers["content-type"], "image/png");        assert(body);        done();      });    });    it("should not fail on simultaneous requests", function(done) {      var url = "http://localhost:3000/avatars/696a82ce41f44b51aa31b8709b8686f0";      // 10 requests at once      var requests = 10;      var finished = 0;      function partDone() {        finished++;        if (requests === finished) {          done();        }      }      function req() {        request.get(url, function(error, res, body) {          assert.ifError(error);          assert.strictEqual(res.statusCode, 200);          assert_headers(res);          assert(res.headers.etag);          assert.strictEqual(res.headers["content-type"], "image/png");          assert(body);          partDone();        });      }      // make simultanous requests      for (var j = 0; j < requests; j++) {        req(j);      }    });    var server_tests = {      "avatar with existing username": {        url: "http://localhost:3000/avatars/jeb_?size=16",        etag: '"a846b82963"',        crc32: 1623808067      },      "avatar with non-existent username": {        url: "http://localhost:3000/avatars/0?size=16",        etag: '"steve"',        crc32: [2416827277, 1243826040]      },      "avatar with non-existent username defaulting to alex": {        url: "http://localhost:3000/avatars/0?size=16&default=alex",        etag: '"alex"',        crc32: [862751081, 809395677]      },      "avatar with non-existent username defaulting to username": {        url: "http://localhost:3000/avatars/0?size=16&default=jeb_",        crc32: 0,        redirect: "/avatars/jeb_?size=16"      },      "avatar with non-existent username defaulting to uuid": {        url: "http://localhost:3000/avatars/0?size=16&default=853c80ef3c3749fdaa49938b674adae6",        crc32: 0,        redirect: "/avatars/853c80ef3c3749fdaa49938b674adae6?size=16"      },      "avatar with non-existent username defaulting to url": {        url: "http://localhost:3000/avatars/0?size=16&default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "helm avatar with existing username": {        url: "http://localhost:3000/avatars/jeb_?size=16&helm",        etag: '"a846b82963"',        crc32: 646871998      },      "helm avatar with non-existent username": {        url: "http://localhost:3000/avatars/0?size=16&helm",        etag: '"steve"',        crc32: [2416827277, 1243826040]      },      "helm avatar with non-existent username defaulting to alex": {        url: "http://localhost:3000/avatars/0?size=16&helm&default=alex",        etag: '"alex"',        crc32: [862751081, 809395677]      },      "helm avatar with non-existent username defaulting to username": {        url: "http://localhost:3000/avatars/0?size=16&helm&default=jeb_",        crc32: 0,        redirect: "/avatars/jeb_?size=16&helm="      },      "helm avatar with non-existent username defaulting to uuid": {        url: "http://localhost:3000/avatars/0?size=16&helm&default=853c80ef3c3749fdaa49938b674adae6",        crc32: 0,        redirect: "/avatars/853c80ef3c3749fdaa49938b674adae6?size=16&helm="      },      "helm avatar with non-existent username defaulting to url": {        url: "http://localhost:3000/avatars/0?size=16&helm&default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "avatar with existing uuid": {        url: "http://localhost:3000/avatars/853c80ef3c3749fdaa49938b674adae6?size=16",        etag: '"a846b82963"',        crc32: 1623808067      },      "avatar with non-existent uuid": {        url: "http://localhost:3000/avatars/00000000000000000000000000000000?size=16",        etag: '"steve"',        crc32: [2416827277, 1243826040]      },      "avatar with non-existent uuid defaulting to alex": {        url: "http://localhost:3000/avatars/00000000000000000000000000000000?size=16&default=alex",        etag: '"alex"',        crc32: [862751081, 809395677]      },      "avatar with non-existent uuid defaulting to username": {        url: "http://localhost:3000/avatars/00000000000000000000000000000000?size=16&default=jeb_",        crc32: 0,        redirect: "/avatars/jeb_?size=16"      },      "avatar with non-existent uuid defaulting to uuid": {        url: "http://localhost:3000/avatars/00000000000000000000000000000000?size=16&default=853c80ef3c3749fdaa49938b674adae6",        crc32: 0,        redirect: "/avatars/853c80ef3c3749fdaa49938b674adae6?size=16"      },      "avatar with non-existent uuid defaulting to url": {        url: "http://localhost:3000/avatars/00000000000000000000000000000000?size=16&default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "helm avatar with existing uuid": {        url: "http://localhost:3000/avatars/853c80ef3c3749fdaa49938b674adae6?size=16&helm",        etag: '"a846b82963"',        crc32: 646871998      },      "helm avatar with non-existent uuid": {        url: "http://localhost:3000/avatars/00000000000000000000000000000000?size=16&helm",        etag: '"steve"',        crc32: [2416827277, 1243826040]      },      "helm avatar with non-existent uuid defaulting to alex": {        url: "http://localhost:3000/avatars/00000000000000000000000000000000?size=16&helm&default=alex",        etag: '"alex"',        crc32: [862751081, 809395677]      },      "helm avatar with non-existent uuid defaulting to username": {        url: "http://localhost:3000/avatars/00000000000000000000000000000000?size=16&default=jeb_",        crc32: 0,        redirect: "/avatars/jeb_?size=16"      },      "helm avatar with non-existent uuid defaulting to uuid": {        url: "http://localhost:3000/avatars/00000000000000000000000000000000?size=16&default=853c80ef3c3749fdaa49938b674adae6",        crc32: 0,        redirect: "/avatars/853c80ef3c3749fdaa49938b674adae6?size=16"      },      "helm avatar with non-existent uuid defaulting to url": {        url: "http://localhost:3000/avatars/00000000000000000000000000000000?size=16&helm&default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "cape with existing username": {        url: "http://localhost:3000/capes/jeb_",        etag: '"3f688e0e69"',        crc32: [989800403, 1901140141]      },      "cape with non-existent username": {        url: "http://localhost:3000/capes/0",        crc32: 0      },      "cape with non-existent username defaulting to url": {        url: "http://localhost:3000/capes/0?default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "cape with existing uuid": {        url: "http://localhost:3000/capes/853c80ef3c3749fdaa49938b674adae6",        etag: '"3f688e0e69"',        crc32: [989800403, 1901140141]      },      "cape with non-existent uuid": {        url: "http://localhost:3000/capes/00000000000000000000000000000000",        crc32: 0      },      "cape with non-existent uuid defaulting to url": {        url: "http://localhost:3000/capes/00000000000000000000000000000000?default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "skin with existing username": {        url: "http://localhost:3000/skins/jeb_",        etag: '"a846b82963"',        crc32: 26500336      },      "skin with non-existent username": {        url: "http://localhost:3000/skins/0",        etag: '"steve"',        crc32: 981937087      },      "skin with non-existent username defaulting to alex": {        url: "http://localhost:3000/skins/0?default=alex",        etag: '"alex"',        crc32: 2298915739      },      "skin with non-existent username defaulting to username": {        url: "http://localhost:3000/skins/0?size=16&default=jeb_",        crc32: 0,        redirect: "/skins/jeb_?size=16"      },      "skin with non-existent username defaulting to uuid": {        url: "http://localhost:3000/skins/0?size=16&default=853c80ef3c3749fdaa49938b674adae6",        crc32: 0,        redirect: "/skins/853c80ef3c3749fdaa49938b674adae6?size=16"      },      "skin with non-existent username defaulting to url": {        url: "http://localhost:3000/skins/0?default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "skin with existing uuid": {        url: "http://localhost:3000/skins/853c80ef3c3749fdaa49938b674adae6",        etag: '"a846b82963"',        crc32: 26500336      },      "skin with non-existent uuid": {        url: "http://localhost:3000/skins/00000000000000000000000000000000",        etag: '"steve"',        crc32: 981937087      },      "skin with non-existent uuid defaulting to alex": {        url: "http://localhost:3000/skins/00000000000000000000000000000000?default=alex",        etag: '"alex"',        crc32: 2298915739      },      "skin with non-existent uuid defaulting to username": {        url: "http://localhost:3000/skins/00000000000000000000000000000000?size=16&default=jeb_",        crc32: 0,        redirect: "/skins/jeb_?size=16"      },      "skin with non-existent uuid defaulting to uuid": {        url: "http://localhost:3000/skins/00000000000000000000000000000000?size=16&default=853c80ef3c3749fdaa49938b674adae6",        crc32: 0,        redirect: "/skins/853c80ef3c3749fdaa49938b674adae6?size=16"      },      "skin with non-existent uuid defaulting to url": {        url: "http://localhost:3000/skins/00000000000000000000000000000000?default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "head render with existing username": {        url: "http://localhost:3000/renders/head/jeb_?scale=2",        etag: '"a846b82963"',        crc32: [353633671, 370672768]      },      "head render with non-existent username": {        url: "http://localhost:3000/renders/head/0?scale=2",        etag: '"steve"',        crc32: [883439147, 433083528]      },      "head render with non-existent username defaulting to alex": {        url: "http://localhost:3000/renders/head/0?scale=2&default=alex",        etag: '"alex"',        crc32: [1240086237, 1108800327]      },      "head render with non-existent username defaulting to username": {        url: "http://localhost:3000/avatars/0?scale=2&default=jeb_",        crc32: 0,        redirect: "/avatars/jeb_?scale=2"      },      "head render with non-existent username defaulting to uuid": {        url: "http://localhost:3000/avatars/0?scale=2&default=853c80ef3c3749fdaa49938b674adae6",        crc32: 0,        redirect: "/avatars/853c80ef3c3749fdaa49938b674adae6?scale=2"      },      "head render with non-existent username defaulting to url": {        url: "http://localhost:3000/renders/head/0?scale=2&default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "helm head render with existing username": {        url: "http://localhost:3000/renders/head/jeb_?scale=2&helm",        etag: '"a846b82963"',        crc32: [3456497067, 3490318764]      },      "helm head render with non-existent username": {        url: "http://localhost:3000/renders/head/0?scale=2&helm",        etag: '"steve"',        crc32: [1858563554, 2647471936]      },      "helm head render with non-existent username defaulting to alex": {        url: "http://localhost:3000/renders/head/0?scale=2&helm&default=alex",        etag: '"alex"',        crc32: [2963161105, 1769904825]      },      "helm head render with non-existent username defaulting to username": {        url: "http://localhost:3000/renders/head/0?scale=2&helm&default=jeb_",        crc32: 0,        redirect: "/renders/head/jeb_?scale=2&helm="      },      "helm head render with non-existent username defaulting to uuid": {        url: "http://localhost:3000/renders/head/0?scale=2&helm&default=853c80ef3c3749fdaa49938b674adae6",        crc32: 0,        redirect: "/renders/head/853c80ef3c3749fdaa49938b674adae6?scale=2&helm="      },      "helm head render with non-existent username defaulting to url": {        url: "http://localhost:3000/renders/head/0?scale=2&helm&default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "head render with existing uuid": {        url: "http://localhost:3000/renders/head/853c80ef3c3749fdaa49938b674adae6?scale=2",        etag: '"a846b82963"',        crc32: [353633671, 370672768]      },      "head render with non-existent uuid": {        url: "http://localhost:3000/renders/head/00000000000000000000000000000000?scale=2",        etag: '"steve"',        crc32: [883439147, 433083528]      },      "head render with non-existent uuid defaulting to alex": {        url: "http://localhost:3000/renders/head/00000000000000000000000000000000?scale=2&default=alex",        etag: '"alex"',        crc32: [1240086237, 1108800327]      },      "head render with non-existent uuid defaulting to username": {        url: "http://localhost:3000/renders/head/00000000000000000000000000000000?scale=2&default=jeb_",        crc32: 0,        redirect: "/renders/head/jeb_?scale=2"      },      "head render with non-existent uuid defaulting to uuid": {        url: "http://localhost:3000/renders/head/00000000000000000000000000000000?scale=2&default=853c80ef3c3749fdaa49938b674adae6",        crc32: 0,        redirect: "/renders/head/853c80ef3c3749fdaa49938b674adae6?scale=2"      },      "head render with non-existent uuid defaulting to url": {        url: "http://localhost:3000/renders/head/00000000000000000000000000000000?scale=2&default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "helm head render with existing uuid": {        url: "http://localhost:3000/renders/head/853c80ef3c3749fdaa49938b674adae6?scale=2&helm",        etag: '"a846b82963"',        crc32: [3456497067, 3490318764]      },      "helm head render with non-existent uuid": {        url: "http://localhost:3000/renders/head/00000000000000000000000000000000?scale=2&helm",        etag: '"steve"',        crc32: [1858563554, 2647471936]      },      "helm head render with non-existent uuid defaulting to alex": {        url: "http://localhost:3000/renders/head/00000000000000000000000000000000?scale=2&helm&default=alex",        etag: '"alex"',        crc32: [2963161105, 1769904825]      },      "helm head with non-existent uuid defaulting to username": {        url: "http://localhost:3000/renders/head/00000000000000000000000000000000?scale=2&helm&default=jeb_",        crc32: 0,        redirect: "/renders/head/jeb_?scale=2&helm="      },      "helm head with non-existent uuid defaulting to uuid": {        url: "http://localhost:3000/renders/head/00000000000000000000000000000000?scale=2&helm&default=853c80ef3c3749fdaa49938b674adae6",        crc32: 0,        redirect: "/renders/head/853c80ef3c3749fdaa49938b674adae6?scale=2&helm="      },      "helm head render with non-existent uuid defaulting to url": {        url: "http://localhost:3000/renders/head/00000000000000000000000000000000?scale=2&helm&default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "body render with existing username": {        url: "http://localhost:3000/renders/body/jeb_?scale=2",        etag: '"a846b82963"',        crc32: [1291941229, 2628108474]      },      "body render with non-existent username": {        url: "http://localhost:3000/renders/body/0?scale=2",        etag: '"steve"',        crc32: [2652947188, 2115706574]      },      "body render with non-existent username defaulting to alex": {        url: "http://localhost:3000/renders/body/0?scale=2&default=alex",        etag: '"alex"',        crc32: [407932087, 2516216042]      },      "body render with non-existent username defaulting to username": {        url: "http://localhost:3000/renders/body/0?scale=2&default=jeb_",        crc32: 0,        redirect: "/renders/body/jeb_?scale=2"      },      "body render with non-existent username defaulting to uuid": {        url: "http://localhost:3000/renders/body/0?scale=2&default=853c80ef3c3749fdaa49938b674adae6",        crc32: 0,        redirect: "/renders/body/853c80ef3c3749fdaa49938b674adae6?scale=2"      },      "body render with non-existent username defaulting to url": {        url: "http://localhost:3000/renders/body/0?scale=2&default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "helm body render with existing username": {        url: "http://localhost:3000/renders/body/jeb_?scale=2&helm",        etag: '"a846b82963"',        crc32: [3556188297, 4269754007]      },      "helm body render with non-existent username": {        url: "http://localhost:3000/renders/body/0?scale=2&helm",        etag: '"steve"',        crc32: [272191039, 542896675]      },      "helm body render with non-existent username defaulting to alex": {        url: "http://localhost:3000/renders/body/0?scale=2&helm&default=alex",        etag: '"alex"',        crc32: [737759773, 66512449]      },      "helm body render with non-existent username defaulting to username": {        url: "http://localhost:3000/renders/body/0?scale=2&helm&default=jeb_",        crc32: 0,        redirect: "/renders/body/jeb_?scale=2&helm="      },      "helm body render with non-existent username defaulting to uuid": {        url: "http://localhost:3000/renders/body/0?scale=2&helm&default=853c80ef3c3749fdaa49938b674adae6",        crc32: 0,        redirect: "/renders/body/853c80ef3c3749fdaa49938b674adae6?scale=2&helm="      },      "helm body render with non-existent username defaulting to url": {        url: "http://localhost:3000/renders/body/0?scale=2&helm&default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "body render with existing uuid": {        url: "http://localhost:3000/renders/body/853c80ef3c3749fdaa49938b674adae6?scale=2",        etag: '"a846b82963"',        crc32: [1291941229, 2628108474]      },      "body render with non-existent uuid": {        url: "http://localhost:3000/renders/body/00000000000000000000000000000000?scale=2",        etag: '"steve"',        crc32: [2652947188, 2115706574]      },      "body render with non-existent uuid defaulting to alex": {        url: "http://localhost:3000/renders/body/00000000000000000000000000000000?scale=2&default=alex",        etag: '"alex"',        crc32: [407932087, 2516216042]      },      "body render with non-existent uuid defaulting to username": {        url: "http://localhost:3000/renders/body/0?scale=2&default=jeb_",        crc32: 0,        redirect: "/renders/body/jeb_?scale=2"      },      "body render with non-existent uuid defaulting to uuid": {        url: "http://localhost:3000/renders/body/0?scale=2&default=853c80ef3c3749fdaa49938b674adae6",        crc32: 0,        redirect: "/renders/body/853c80ef3c3749fdaa49938b674adae6?scale=2"      },      "body render with non-existent uuid defaulting to url": {        url: "http://localhost:3000/renders/body/00000000000000000000000000000000?scale=2&default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },      "helm body render with existing uuid": {        url: "http://localhost:3000/renders/body/853c80ef3c3749fdaa49938b674adae6?scale=2&helm",        etag: '"a846b82963"',        crc32: [3556188297, 4269754007]      },      "helm body render with non-existent uuid": {        url: "http://localhost:3000/renders/body/00000000000000000000000000000000?scale=2&helm",        etag: '"steve"',        crc32: [272191039, 542896675]      },      "helm body render with non-existent uuid defaulting to alex": {        url: "http://localhost:3000/renders/body/00000000000000000000000000000000?scale=2&helm&default=alex",        etag: '"alex"',        crc32: [737759773, 66512449]      },      "helm body render with non-existent uuid defaulting to url": {        url: "http://localhost:3000/renders/body/00000000000000000000000000000000?scale=2&helm&default=http%3A%2F%2Fexample.com",        crc32: 0,        redirect: "http://example.com"      },    };    for (var description in server_tests) {      var loc = server_tests[description];      (function(location) {        it("should return correct HTTP response for " + description, function(done) {          request.get(location.url, {followRedirect: false, encoding: null}, function(error, res, body) {            assert.ifError(error);            assert_headers(res);            assert(res.headers["x-storage-type"]);            assert.strictEqual(res.headers.etag, location.etag);            var matches = false;            if (location.crc32 instanceof Array) {              for (var i = 0; i < location.crc32.length; i++) {                if (location.crc32[i] === crc(body)) {                  matches = true;                  break;                }              }            } else {              matches = (location.crc32 === crc(body));            }            try {              assert.ok(matches);            } catch(e) {              throw new Error(crc(body) + " != " + location.crc32);            }            assert.strictEqual(res.headers.location, location.redirect);            if (location.etag === undefined) {              assert.strictEqual(res.statusCode, location.redirect ? 307 : 404);              assert.strictEqual(res.headers["content-type"], "text/plain");              done();            } else {              assert(res.headers.etag);              assert.strictEqual(res.headers["content-type"], "image/png");              assert.strictEqual(res.statusCode, 200);              assert_cache(location.url, res.headers.etag, function() {                done();              });            }          });        });      }(loc));    }    it("should return a 422 (invalid size)", function(done) {      var size = config.avatars.max_size + 1;      request.get("http://localhost:3000/avatars/Jake_0?size=" + size, function(error, res, body) {        assert.ifError(error);        assert.strictEqual(res.statusCode, 422);        done();      });    });    it("should return a 422 (invalid scale)", function(done) {      var scale = config.renders.max_scale + 1;      request.get("http://localhost:3000/renders/head/Jake_0?scale=" + scale, function(error, res, body) {        assert.ifError(error);        assert.strictEqual(res.statusCode, 422);        done();      });    });    it("should return a 422 (invalid render type)", function(done) {      request.get("http://localhost:3000/renders/invalid/Jake_0", function(error, res, body) {        assert.ifError(error);        assert.strictEqual(res.statusCode, 422);        done();      });    });    // testing all paths for Invalid UserID    var locations = ["avatars", "skins", "capes", "renders/body", "renders/head"];    for (var l in locations) {      loc = locations[l];      (function(location) {        it("should return a 422 (invalid id " + location + ")", function(done) {          request.get("http://localhost:3000/" + location + "/thisisaninvaliduuid", function(error, res, body) {            assert.ifError(error);            assert.strictEqual(res.statusCode, 422);            done();          });        });        it("should return a 422 (invalid request path " + location + ")", function(done) {          request.get("http://localhost:3000/" + location + "/thisisaninvaliduuid/invalid", function(error, res, body) {            assert.ifError(error);            assert.strictEqual(res.statusCode, 422);            done();          });        });      }(loc));    }    after(function(done) {      server.close(function() {        done();      });    });  });  // we have to make sure that we test both a 32x64 and 64x64 skin  describe("Networking: Render", function() {    it("should not fail (username, 32x64 skin)", function(done) {      helpers.get_render(rid, "md_5", 6, true, true, function(err, hash, img) {        assert.strictEqual(err, null);        done();      });    });    it("should not fail (username, 64x64 skin)", function(done) {      helpers.get_render(rid, "Jake_0", 6, true, true, function(err, hash, img) {        assert.strictEqual(err, null);        done();      });    });  });  describe("Networking: Cape", function() {    it("should not fail (guaranteed cape)", function(done) {      helpers.get_cape(rid, "Dinnerbone", function(err, hash, status, img) {        assert.strictEqual(err, null);        done();      });    });    it("should already exist", function(done) {      before(function() {        cache.get_redis().flushall();      });      helpers.get_cape(rid, "Dinnerbone", function(err, hash, status, img) {        assert.strictEqual(err, null);        done();      });    });    it("should not be found", function(done) {      helpers.get_cape(rid, "Jake_0", function(err, hash, status, img) {        assert.ifError(err);        assert.strictEqual(img, null);        done();      });    });  });  describe("Networking: Skin", function() {    it("should not fail", function(done) {      helpers.get_cape(rid, "Jake_0", function(err, hash, status, img) {        assert.strictEqual(err, null);        done();      });    });    it("should already exist", function(done) {      before(function() {        cache.get_redis().flushall();      });      helpers.get_cape(rid, "Jake_0", function(err, hash, status, img) {        assert.strictEqual(err, null);        done();      });    });  });  // DRY with uuid and username tests  for (var i in ids) {    var iid = ids[i];    var iid_type = iid.length > 16 ? "uuid" : "name";    // needs an anonymous function because id and id_type aren't constant    (function(id, id_type) {      describe("Networking: Avatar", function() {        before(function() {          cache.get_redis().flushall();          console.log("\n\nRunning tests with " + id_type + " '" + id + "'\n\n");        });        it("should be downloaded", function(done) {          helpers.get_avatar(rid, id, false, 160, function(err, status, image) {            assert.ifError(err);            assert.strictEqual(status, 2);            done();          });        });        it("should be cached", function(done) {          helpers.get_avatar(rid, id, false, 160, function(err, status, image) {            assert.ifError(err);            assert.strictEqual(status === 0 || status === 1, true);            done();          });        });        if (id.length > 16) {          console.log("can't run 'checked' test due to Mojang's rate limits :(");        } else {          it("should be checked", function(done) {            var original_cache_time = config.caching.local;            config.caching.local = 0;            helpers.get_avatar(rid, id, false, 160, function(err, status, image) {              assert.ifError(err);              assert.strictEqual(status, 3);              config.caching.local = original_cache_time;              done();            });          });        }      });      describe("Networking: Skin", function() {        it("should not fail (uuid)", function(done) {          helpers.get_skin(rid, id, function(err, hash, status, img) {            assert.strictEqual(err, null);            done();          });        });      });      describe("Networking: Render", function() {        it("should not fail (full body)", function(done) {          helpers.get_render(rid, id, 6, true, true, function(err, hash, img) {            assert.ifError(err);            done();          });        });        it("should not fail (only head)", function(done) {          helpers.get_render(rid, id, 6, true, false, function(err, hash, img) {            assert.ifError(err);            done();          });        });      });      describe("Networking: Cape", function() {        it("should not fail (possible cape)", function(done) {          helpers.get_cape(rid, id, function(err, hash, status, img) {            assert.ifError(err);            done();          });        });      });      describe("Errors", function() {        before(function() {          cache.get_redis().flushall();        });        if (id_type === "uuid") {          it("uuid should be rate limited", function(done) {            networking.get_profile(rid, id, function() {              networking.get_profile(rid, id, function(err, profile) {                assert.strictEqual(err, "TooManyRequests");                assert.strictEqual(profile.error, "TooManyRequestsException");                done();              });            });          });        } else {          it("username should NOT be rate limited (username)", function(done) {            helpers.get_avatar(rid, id, false, 160, function() {              helpers.get_avatar(rid, id, false, 160, function(err, status, image) {                assert.strictEqual(err, null);                done();              });            });          });        }      });    }(iid, iid_type));  }});
 |