client/preprocessing/api.js

  1. var linkedList = require('../../common/linkedlist.js');
  2. module.exports = function (jiffClient) {
  3. var isRunning = false;
  4. var userCallbacks = [];
  5. var preprocessingTasks = [linkedList()];
  6. /**
  7. * Checks if the given operation uses preprocessed values
  8. * @method has_preprocessing
  9. * @memberof module:jiff-client~JIFFClient
  10. * @instance
  11. * @param {string} op - name of the operation to check
  12. * @return {boolean} true if the op uses preprocessing, false otherwise
  13. */
  14. jiffClient.has_preprocessing = function (op) {
  15. for (var i = 0; i < jiffClient.extensions.length; i++) {
  16. if (jiffClient.preprocessing_function_map[jiffClient.extensions[i]][op] != null) {
  17. return true;
  18. }
  19. }
  20. return false;
  21. };
  22. /**
  23. * Get a preprocessed share/value by associated op_id. If value does not exist
  24. * Fallback to some user specified way for creating it
  25. * @method get_preprocessing
  26. * @memberof module:jiff-client~JIFFClient
  27. * @instance
  28. * @param {string} op_id - the op_id associated with the preprocessed value/share
  29. * @return {object} the preprocessed share(s)
  30. */
  31. jiffClient.get_preprocessing = function (op_id) {
  32. var values = jiffClient.preprocessing_table[op_id];
  33. if (values != null) {
  34. return values;
  35. }
  36. if (jiffClient.crypto_provider === true) {
  37. return null;
  38. }
  39. throw new Error('No preprocessed value(s) that correspond to the op_id "' + op_id + '"');
  40. };
  41. /**
  42. * Store a pair of op_id and associated pre-processed value/share
  43. * The value/share can be accessed later during the computation through jiffClient.get_preprocessing(op_id)
  44. * @method store_preprocessing
  45. * @memberof module:jiff-client~JIFFClient
  46. * @instance
  47. * @param {string} op_id - the op_id associated with the preprocessed value/share
  48. * @param {SecretShare} share - the share/value to store
  49. */
  50. jiffClient.store_preprocessing = function (op_id, share) {
  51. if (share != null) {
  52. jiffClient.preprocessing_table[op_id] = share;
  53. }
  54. };
  55. /**
  56. * Generate values used for JIFF operations in advance of the computation.
  57. *
  58. * Calling this function does not begin preprocessing, it just creates a preprocessing task. After you created
  59. * your desired tasks, you can ask JIFF to execute them via {@link executePreprocessing}.
  60. *
  61. * @method preprocessing
  62. * @memberof module:jiff-client~JIFFClient
  63. * @instance
  64. * @param {string} dependent_op - name of the operation that will later use the pre_processed values
  65. * @param {Number} [count=1] - number of times the protocol should be performed, number of values that will be generated
  66. * @param {Object} [protocols=defaults] - a mapping from base preprocessing elements ('beaver', 'bits', 'sampling') to functions that can pre-process them
  67. * the function must implement the same interface as the JIFF provided protocols (e.g. jiffClient.protocols.generate_beaver_bgw),
  68. * missing mappings indicate that JIFF must use the default protocols
  69. * @param {Number} [threshold=receivers_list.length] - the threshold of the preprocessed shares
  70. * @param {Array} [receivers_list=all_parties] - the parties that will receive the preprocsssed shares
  71. * @param {Array} [compute_list=all_parties] - the parties that will compute the preprocsssed shares
  72. * @param {Number} [Zp=jiffClient.Zp] - the Zp of the preprocessed shares
  73. * @param {Array} [id_list=auto_gen()] - array of ids to be used sequentially to identify the pre_processed values
  74. * @param {Object} [params={}] - any additional protocol-specific parameters
  75. * @return {promise} a promise that is resolved when preprocessing is completed, null if this is called by a party that is neither a compute nor receiver party
  76. * @see {@link executePreprocessing}
  77. */
  78. jiffClient.preprocessing = function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, id_list, params) {
  79. // defaults!
  80. if (receivers_list == null) {
  81. receivers_list = [];
  82. for (var p = 1; p <= jiffClient.party_count; p++) {
  83. receivers_list.push(p);
  84. }
  85. } else {
  86. jiffClient.helpers.sort_ids(receivers_list);
  87. }
  88. if (compute_list == null) {
  89. compute_list = [];
  90. for (var c = 1; c <= jiffClient.party_count; c++) {
  91. compute_list.push(c);
  92. }
  93. } else {
  94. jiffClient.helpers.sort_ids(compute_list);
  95. }
  96. // not a receiver nor a sender
  97. if (receivers_list.indexOf(jiffClient.id) === -1 && compute_list.indexOf(jiffClient.id) === -1) {
  98. return null;
  99. }
  100. // more defaults
  101. if (Zp == null) {
  102. Zp = jiffClient.Zp;
  103. }
  104. if (threshold == null) {
  105. threshold = receivers_list.length;
  106. }
  107. if (protocols == null) {
  108. protocols = jiffClient.default_preprocessing_protocols;
  109. }
  110. // actual preprocessing
  111. if (count == null || count <= 0) {
  112. count = 1;
  113. }
  114. if (params == null) {
  115. params = {};
  116. }
  117. if (params['namespace'] == null) {
  118. params['namespace'] = jiffClient.extensions[jiffClient.extensions.length - 1];
  119. }
  120. // Create preprocessing tasks
  121. var task = {
  122. dependent_op : dependent_op,
  123. count : count,
  124. threshold : threshold,
  125. receivers_list : receivers_list,
  126. compute_list : compute_list,
  127. Zp : Zp,
  128. id_list : id_list,
  129. id : null,
  130. params : params,
  131. protocols : protocols,
  132. deferred: new jiffClient.helpers.Deferred()
  133. };
  134. preprocessingTasks[preprocessingTasks.length - 1].add(task);
  135. return task.deferred.promise;
  136. };
  137. /**
  138. * Ask JIFF to start executing preprocessing for tasks previously added by {@link preprocessing}.
  139. *
  140. * Calls the provided callback when the preprocessing tasks are all done.
  141. *
  142. * @method executePreprocessing
  143. * @memberof module:jiff-client~JIFFClient
  144. * @instance
  145. * @param callback {!Function} - the callback to execute when preprocessing is finished.
  146. * {@link preprocessing}
  147. */
  148. jiffClient.executePreprocessing = function (callback) {
  149. userCallbacks.push(callback);
  150. preprocessingTasks.push(linkedList());
  151. if (!isRunning) {
  152. __executePreprocessing();
  153. }
  154. };
  155. // called only when preprocessing can run RIGHT NOW
  156. var __executePreprocessing = function () {
  157. isRunning = true;
  158. jiffClient.currentPreprocessingTasks = preprocessingTasks.shift();
  159. var currentCallback = userCallbacks.shift();
  160. jiffClient.preprocessingCallback = function () {
  161. if (currentCallback != null) {
  162. currentCallback.apply(null, currentCallback);
  163. }
  164. if (userCallbacks.length > 0) {
  165. __executePreprocessing();
  166. } else {
  167. isRunning = false;
  168. }
  169. };
  170. jiffClient.preprocessingDaemon();
  171. };
  172. };