helpers.js 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. var networking = require("./networking");
  2. var logging = require("./logging");
  3. var config = require("./config");
  4. var cache = require("./cache");
  5. var skins = require("./skins");
  6. var fs = require("fs");
  7. // 0098cb60-fa8e-427c-b299-793cbd302c9a
  8. var valid_uuid = /^([0-9a-f-A-F-]{32,36}|[a-zA-Z0-9_]{1,16})$/; // uuid|username
  9. var hash_pattern = /[0-9a-f]+$/;
  10. function get_hash(url) {
  11. return hash_pattern.exec(url)[0].toLowerCase();
  12. }
  13. // requests skin for +uuid+ and extracts face/helm if image hash in +details+ changed
  14. // callback contains error, image hash
  15. function store_images(uuid, details, callback) {
  16. // get skin_url for +uuid+
  17. networking.get_skin_url(uuid, function(err, skin_url) {
  18. if (err) {
  19. callback(err, null);
  20. } else {
  21. if (skin_url) {
  22. logging.log(uuid + " " + skin_url);
  23. // set file paths
  24. var hash = get_hash(skin_url);
  25. if (details && details.hash == hash) {
  26. // hash hasn't changed
  27. logging.log(uuid + " hash has not changed");
  28. cache.update_timestamp(uuid, hash);
  29. callback(null, hash);
  30. } else {
  31. // hash has changed
  32. logging.log(uuid + " new hash: " + hash);
  33. var facepath = __dirname + "/../" + config.faces_dir + hash + ".png";
  34. var helmpath = __dirname + "/../" + config.helms_dir + hash + ".png";
  35. if (fs.existsSync(facepath)) {
  36. logging.log(uuid + " Avatar already exists, not downloading");
  37. cache.save_hash(uuid, hash);
  38. callback(null, hash);
  39. } else {
  40. // download skin
  41. networking.get_skin(skin_url, function(err, img) {
  42. if (err || !img) {
  43. callback(err, null);
  44. } else {
  45. // extract face / helm
  46. skins.extract_face(img, facepath, function(err) {
  47. if (err) {
  48. callback(err);
  49. } else {
  50. logging.log(uuid + " face extracted");
  51. logging.debug(facepath);
  52. skins.extract_helm(facepath, img, helmpath, function(err) {
  53. logging.log(uuid + " helm extracted");
  54. logging.debug(helmpath);
  55. cache.save_hash(uuid, hash);
  56. callback(err, hash);
  57. });
  58. }
  59. });
  60. }
  61. });
  62. }
  63. }
  64. } else {
  65. // profile found, but has no skin
  66. cache.save_hash(uuid, null);
  67. callback(null, null);
  68. }
  69. }
  70. });
  71. }
  72. var exp = {};
  73. // returns true if the +uuid+ is a valid uuid or username
  74. // the uuid may be not exist, however
  75. exp.uuid_valid = function(uuid) {
  76. return valid_uuid.test(uuid);
  77. };
  78. // decides whether to get an image from disk or to download it
  79. // callback contains error, status, hash
  80. // the status gives information about how the image was received
  81. // -1: "error"
  82. // 0: "none" - cached as null
  83. // 1: "cached" - found on disk
  84. // 2: "downloaded" - profile downloaded, skin downloaded from mojang servers
  85. // 3: "checked" - profile re-downloaded (was too old), but it has either not changed or has no skin
  86. exp.get_image_hash = function(uuid, callback) {
  87. cache.get_details(uuid, function(err, details) {
  88. if (err) {
  89. callback(err, -1, null);
  90. } else {
  91. if (details && details.time + config.local_cache_time * 1000 >= new Date().getTime()) {
  92. // uuid known + recently updated
  93. logging.log(uuid + " uuid cached & recently updated");
  94. callback(null, (details.hash ? 1 : 0), details.hash);
  95. } else {
  96. if (details) {
  97. logging.log(uuid + " uuid cached, but too old");
  98. } else {
  99. logging.log(uuid + " uuid not cached");
  100. }
  101. store_images(uuid, details, function(err, hash) {
  102. if (err) {
  103. callback(err, -1, details && details.hash);
  104. } else {
  105. // skin is only checked (3) when uuid known AND hash didn't change
  106. // in all other cases the skin is downloaded (2)
  107. var status = details && (details.hash == hash) ? 3 : 2;
  108. logging.debug(uuid + " old hash: " + (details && details.hash));
  109. logging.log(uuid + " hash: " + hash);
  110. callback(null, status, hash);
  111. }
  112. });
  113. }
  114. }
  115. });
  116. };
  117. // handles requests for +uuid+ avatars with +size+
  118. // callback contains error, status, image buffer, hash
  119. // image is the user's face+helm when helm is true, or the face otherwise
  120. // for status, see get_image_hash
  121. exp.get_avatar = function(uuid, helm, size, callback) {
  122. logging.log("\nrequest: " + uuid);
  123. exp.get_image_hash(uuid, function(err, status, hash) {
  124. if (hash) {
  125. var facepath = __dirname + "/../" + config.faces_dir + hash + ".png";
  126. var helmpath = __dirname + "/../" + config.helms_dir + hash + ".png";
  127. var filepath = facepath;
  128. if (helm && fs.existsSync(helmpath)) {
  129. filepath = helmpath;
  130. }
  131. skins.resize_img(filepath, size, function(img_err, result) {
  132. if (img_err) {
  133. callback(img_err, -1, null, hash);
  134. } else {
  135. // we might have a hash although an error occured
  136. // (e.g. Mojang servers not reachable, using outdated hash)
  137. callback(err, (err ? -1 : status), result, hash);
  138. }
  139. });
  140. } else {
  141. // hash is null when uuid has no skin
  142. callback(err, status, null, null);
  143. }
  144. });
  145. };
  146. // handles requests for +uuid+ skins
  147. // callback contains error, hash, image buffer
  148. exp.get_skin = function(uuid, callback) {
  149. logging.log(uuid + " skin request");
  150. exp.get_image_hash(uuid, function(err, status, hash) {
  151. if (hash) {
  152. var skinurl = "http://textures.minecraft.net/texture/" + hash;
  153. networking.get_skin(skinurl, function(err, img) {
  154. if (err) {
  155. logging.error("error while downloading skin");
  156. callback(err, hash, null);
  157. } else {
  158. callback(null, hash, img);
  159. }
  160. });
  161. } else {
  162. callback(err, null, null);
  163. }
  164. });
  165. };
  166. module.exports = exp;