skins.js 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. var logging = require("./logging");
  2. var lwip = require("lwip");
  3. var fs = require("fs");
  4. var exp = {};
  5. // extracts the face from an image +buffer+
  6. // result is saved to a file called +outname+
  7. // callback: error
  8. exp.extract_face = function(buffer, outname, callback) {
  9. lwip.open(buffer, "png", function(err, image) {
  10. if (err) {
  11. callback(err);
  12. } else {
  13. image.batch()
  14. .crop(8, 8, 15, 15) // face
  15. .writeFile(outname, function(write_err) {
  16. if (write_err) {
  17. callback(write_err);
  18. } else {
  19. callback(null);
  20. }
  21. });
  22. }
  23. });
  24. };
  25. // extracts the helm from an image +buffer+ and lays it over a +facefile+
  26. // +facefile+ is the filename of an image produced by extract_face
  27. // result is saved to a file called +outname+
  28. // callback: error
  29. exp.extract_helm = function(rid, facefile, buffer, outname, callback) {
  30. lwip.open(buffer, "png", function(err, skin_img) {
  31. if (err) {
  32. callback(err);
  33. } else {
  34. lwip.open(facefile, function(open_err, face_img) {
  35. if (open_err) {
  36. callback(open_err);
  37. } else {
  38. face_img.toBuffer("png", { compression: "none" }, function(buf_err, face_buffer) {
  39. // FIXME: buf_err is not handled
  40. skin_img.crop(40, 8, 47, 15, function(crop_err, helm_img) {
  41. if (crop_err) {
  42. callback(crop_err);
  43. } else {
  44. face_img.paste(0, 0, helm_img, function(img_err, face_helm_img) {
  45. if (img_err) {
  46. callback(img_err);
  47. } else {
  48. face_helm_img.toBuffer("png", {compression: "none"}, function(buf_err2, face_helm_buffer) {
  49. // FIXME: buf_err2 is not handled
  50. if (face_helm_buffer.toString() !== face_buffer.toString()) {
  51. face_helm_img.writeFile(outname, function(write_err) {
  52. callback(write_err);
  53. });
  54. } else {
  55. logging.log(rid, "helm img == face img, not storing!");
  56. callback(null);
  57. }
  58. });
  59. }
  60. });
  61. }
  62. });
  63. });
  64. }
  65. });
  66. }
  67. });
  68. };
  69. // resizes the image file +inname+ to +size+ by +size+ pixels
  70. // callback: error, image buffer
  71. exp.resize_img = function(inname, size, callback) {
  72. lwip.open(inname, function(err, image) {
  73. if (err) {
  74. callback(err, null);
  75. } else {
  76. image.batch()
  77. .resize(size, size, "nearest-neighbor") // nearest-neighbor doesn't blur
  78. .toBuffer("png", function(buf_err, buffer) {
  79. // FIXME: buf_err is not handled
  80. callback(null, buffer);
  81. });
  82. }
  83. });
  84. };
  85. // returns "alex" or "steve" calculated by the +uuid+
  86. exp.default_skin = function(uuid) {
  87. if (uuid.length <= 16) {
  88. // we can't get the skin type by username
  89. return "steve";
  90. } else {
  91. // great thanks to Minecrell for research into Minecraft and Java's UUID hashing!
  92. // https://git.io/xJpV
  93. // MC uses `uuid.hashCode() & 1` for alex
  94. // that can be compacted to counting the LSBs of every 4th byte in the UUID
  95. // an odd sum means alex, an even sum means steve
  96. // XOR-ing all the LSBs gives us 1 for alex and 0 for steve
  97. var lsbs_even = parseInt(uuid[ 7], 16) ^
  98. parseInt(uuid[15], 16) ^
  99. parseInt(uuid[23], 16) ^
  100. parseInt(uuid[31], 16);
  101. return lsbs_even ? "alex" : "steve";
  102. }
  103. };
  104. // helper method for opening a skin file from +skinpath+
  105. // callback: error, image buffer
  106. exp.open_skin = function(rid, skinpath, callback) {
  107. fs.readFile(skinpath, function(err, buf) {
  108. if (err) {
  109. logging.error(rid, "error while opening skin file:", err);
  110. callback(err, null);
  111. } else {
  112. callback(null, buf);
  113. }
  114. });
  115. };
  116. // write the image +buffer+ to the +outpath+ file
  117. // callback: error
  118. exp.save_image = function(buffer, outpath, callback) {
  119. logging.error("outpath:" + outpath);
  120. lwip.open(buffer, "png", function(err, image) {
  121. if (err) {
  122. callback(err);
  123. } else {
  124. image.batch()
  125. .writeFile(outpath, function(write_err) {
  126. if (write_err) {
  127. callback(write_err);
  128. } else {
  129. callback(null);
  130. }
  131. });
  132. }
  133. });
  134. };
  135. module.exports = exp;