helpers.js 7.5 KB

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