binary.js 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. /*global Blob,File*/
  2. /**
  3. * Module requirements
  4. */
  5. var isArray = require('isarray');
  6. var isBuf = require('./is-buffer');
  7. var toString = Object.prototype.toString;
  8. var withNativeBlob = typeof Blob === 'function' || (typeof Blob !== 'undefined' && toString.call(Blob) === '[object BlobConstructor]');
  9. var withNativeFile = typeof File === 'function' || (typeof File !== 'undefined' && toString.call(File) === '[object FileConstructor]');
  10. /**
  11. * Replaces every Buffer | ArrayBuffer in packet with a numbered placeholder.
  12. * Anything with blobs or files should be fed through removeBlobs before coming
  13. * here.
  14. *
  15. * @param {Object} packet - socket.io event packet
  16. * @return {Object} with deconstructed packet and list of buffers
  17. * @api public
  18. */
  19. exports.deconstructPacket = function(packet) {
  20. var buffers = [];
  21. var packetData = packet.data;
  22. var pack = packet;
  23. pack.data = _deconstructPacket(packetData, buffers);
  24. pack.attachments = buffers.length; // number of binary 'attachments'
  25. return {packet: pack, buffers: buffers};
  26. };
  27. function _deconstructPacket(data, buffers) {
  28. if (!data) return data;
  29. if (isBuf(data)) {
  30. var placeholder = { _placeholder: true, num: buffers.length };
  31. buffers.push(data);
  32. return placeholder;
  33. } else if (isArray(data)) {
  34. var newData = new Array(data.length);
  35. for (var i = 0; i < data.length; i++) {
  36. newData[i] = _deconstructPacket(data[i], buffers);
  37. }
  38. return newData;
  39. } else if (typeof data === 'object' && !(data instanceof Date)) {
  40. var newData = {};
  41. for (var key in data) {
  42. newData[key] = _deconstructPacket(data[key], buffers);
  43. }
  44. return newData;
  45. }
  46. return data;
  47. }
  48. /**
  49. * Reconstructs a binary packet from its placeholder packet and buffers
  50. *
  51. * @param {Object} packet - event packet with placeholders
  52. * @param {Array} buffers - binary buffers to put in placeholder positions
  53. * @return {Object} reconstructed packet
  54. * @api public
  55. */
  56. exports.reconstructPacket = function(packet, buffers) {
  57. packet.data = _reconstructPacket(packet.data, buffers);
  58. packet.attachments = undefined; // no longer useful
  59. return packet;
  60. };
  61. function _reconstructPacket(data, buffers) {
  62. if (!data) return data;
  63. if (data && data._placeholder === true) {
  64. var isIndexValid =
  65. typeof data.num === "number" &&
  66. data.num >= 0 &&
  67. data.num < buffers.length;
  68. if (isIndexValid) {
  69. return buffers[data.num]; // appropriate buffer (should be natural order anyway)
  70. } else {
  71. throw new Error("illegal attachments");
  72. }
  73. } else if (isArray(data)) {
  74. for (var i = 0; i < data.length; i++) {
  75. data[i] = _reconstructPacket(data[i], buffers);
  76. }
  77. } else if (typeof data === 'object') {
  78. for (var key in data) {
  79. data[key] = _reconstructPacket(data[key], buffers);
  80. }
  81. }
  82. return data;
  83. }
  84. /**
  85. * Asynchronously removes Blobs or Files from data via
  86. * FileReader's readAsArrayBuffer method. Used before encoding
  87. * data as msgpack. Calls callback with the blobless data.
  88. *
  89. * @param {Object} data
  90. * @param {Function} callback
  91. * @api private
  92. */
  93. exports.removeBlobs = function(data, callback) {
  94. function _removeBlobs(obj, curKey, containingObject) {
  95. if (!obj) return obj;
  96. // convert any blob
  97. if ((withNativeBlob && obj instanceof Blob) ||
  98. (withNativeFile && obj instanceof File)) {
  99. pendingBlobs++;
  100. // async filereader
  101. var fileReader = new FileReader();
  102. fileReader.onload = function() { // this.result == arraybuffer
  103. if (containingObject) {
  104. containingObject[curKey] = this.result;
  105. }
  106. else {
  107. bloblessData = this.result;
  108. }
  109. // if nothing pending its callback time
  110. if(! --pendingBlobs) {
  111. callback(bloblessData);
  112. }
  113. };
  114. fileReader.readAsArrayBuffer(obj); // blob -> arraybuffer
  115. } else if (isArray(obj)) { // handle array
  116. for (var i = 0; i < obj.length; i++) {
  117. _removeBlobs(obj[i], i, obj);
  118. }
  119. } else if (typeof obj === 'object' && !isBuf(obj)) { // and object
  120. for (var key in obj) {
  121. _removeBlobs(obj[key], key, obj);
  122. }
  123. }
  124. }
  125. var pendingBlobs = 0;
  126. var bloblessData = data;
  127. _removeBlobs(bloblessData);
  128. if (!pendingBlobs) {
  129. callback(bloblessData);
  130. }
  131. };