module.exports = { init: function(options) { var req; req = (function(_this) { return function(name, choices) { if (options[name] != null) { if ((choices == null) || choices.some(function(c) { return c === options[name]; })) { return _this[name] = options[name]; } else { throw new Error("You can set the '" + name + "' option to one of the following choices: " + JSON.encode(choices)); } } else { throw new Error("You must specify " + name + ", when initializing the Connector!"); } }; })(this); req("syncMode", ["syncAll", "master-slave"]); req("role", ["master", "slave"]); req("user_id"); this.on_user_id_set(this.user_id); if (this.role === "master") { this.syncMode = "syncAll"; } this.is_synced = false; this.connections = {}; this.is_bound_to_y = false; this.connections = {}; this.current_sync_target = null; return this.sent_hb_to_all_users = false; }, isRoleMaster: function() { return this.role === "master"; }, isRoleSlave: function() { return this.role === "slave"; }, findNewSyncTarget: function() { var c, user, _ref; this.current_sync_target = null; if (this.syncMode === "syncAll") { _ref = this.connections; for (user in _ref) { c = _ref[user]; if (!c.is_synced) { this.performSync(user); break; } } } if (this.current_sync_target == null) { this.setStateSynced(); } return null; }, userLeft: function(user) { delete this.connections[user]; return this.findNewSyncTarget(); }, userJoined: function(user, role) { if (role == null) { throw new Error("Internal: You must specify the role of the joined user! E.g. userJoined('uid:3939','slave')"); } this.connections[user] = { is_synced: false }; if ((!this.is_synced) || this.syncMode === "syncAll") { if (this.syncMode === "syncAll") { return this.performSync(user); } else if (role === "master") { return this.performSyncWithMaster(user); } } }, whenSynced: function(args) { if (args.constructore === Function) { args = [args]; } if (this.is_synced) { return args[0].apply(this, args.slice(1)); } else { if (this.compute_when_synced == null) { this.compute_when_synced = []; } return this.compute_when_synced.push(args); } }, onReceive: function(f) { return this.receive_handlers.push(f); }, /* * Broadcast a message to all connected peers. * @param message {Object} The message to broadcast. * broadcast: (message)-> throw new Error "You must implement broadcast!" * * Send a message to a peer, or set of peers * send: (peer_s, message)-> throw new Error "You must implement send!" */ performSync: function(user) { var hb, o, _hb, _i, _len; if (this.current_sync_target == null) { this.current_sync_target = user; this.send(user, { sync_step: "getHB", send_again: "true", data: [] }); if (!this.sent_hb_to_all_users) { this.sent_hb_to_all_users = true; hb = this.getHB([]).hb; _hb = []; for (_i = 0, _len = hb.length; _i < _len; _i++) { o = hb[_i]; _hb.push(o); if (_hb.length > 30) { this.broadcast({ sync_step: "applyHB_", data: _hb }); _hb = []; } } return this.broadcast({ sync_step: "applyHB", data: _hb }); } } }, performSyncWithMaster: function(user) { var hb, o, _hb, _i, _len; this.current_sync_target = user; this.send(user, { sync_step: "getHB", send_again: "true", data: [] }); hb = this.getHB([]).hb; _hb = []; for (_i = 0, _len = hb.length; _i < _len; _i++) { o = hb[_i]; _hb.push(o); if (_hb.length > 30) { this.broadcast({ sync_step: "applyHB_", data: _hb }); _hb = []; } } return this.broadcast({ sync_step: "applyHB", data: _hb }); }, setStateSynced: function() { var f, _i, _len, _ref; if (!this.is_synced) { this.is_synced = true; _ref = this.compute_when_synced; for (_i = 0, _len = _ref.length; _i < _len; _i++) { f = _ref[_i]; f(); } delete this.compute_when_synced; return null; } }, receiveMessage: function(sender, res) { var data, f, hb, o, sendApplyHB, send_again, _hb, _i, _j, _len, _len1, _ref, _results; if (res.sync_step == null) { _ref = this.receive_handlers; _results = []; for (_i = 0, _len = _ref.length; _i < _len; _i++) { f = _ref[_i]; _results.push(f(sender, res)); } return _results; } else { if (sender === this.user_id) { return; } if (res.sync_step === "getHB") { data = this.getHB(res.data); hb = data.hb; _hb = []; if (this.is_synced) { sendApplyHB = (function(_this) { return function(m) { return _this.send(sender, m); }; })(this); } else { sendApplyHB = (function(_this) { return function(m) { return _this.broadcast(m); }; })(this); } for (_j = 0, _len1 = hb.length; _j < _len1; _j++) { o = hb[_j]; _hb.push(o); if (_hb.length > 30) { sendApplyHB({ sync_step: "applyHB_", data: _hb }); _hb = []; } } sendApplyHB({ sync_step: "applyHB", data: _hb }); if (res.send_again != null) { send_again = (function(_this) { return function(sv) { return function() { hb = _this.getHB(sv).hb; return _this.send(sender, { sync_step: "applyHB", data: hb, sent_again: "true" }); }; }; })(this)(data.state_vector); return setTimeout(send_again, 3000); } } else if (res.sync_step === "applyHB") { this.applyHB(res.data, sender === this.current_sync_target); if ((this.syncMode === "syncAll" || (res.sent_again != null)) && (!this.is_synced) && (this.current_sync_target === sender)) { this.connections[sender].is_synced = true; return this.findNewSyncTarget(); } } else if (res.sync_step === "applyHB_") { return this.applyHB(res.data, sender === this.current_sync_target); } } }, parseMessageFromXml: function(m) { var parse_array, parse_object; parse_array = function(node) { var n, _i, _len, _ref, _results; _ref = node.children; _results = []; for (_i = 0, _len = _ref.length; _i < _len; _i++) { n = _ref[_i]; if (n.getAttribute("isArray") === "true") { _results.push(parse_array(n)); } else { _results.push(parse_object(n)); } } return _results; }; parse_object = function(node) { var int, json, n, name, value, _i, _len, _ref, _ref1; json = {}; _ref = node.attrs; for (name in _ref) { value = _ref[name]; int = parseInt(value); if (isNaN(int) || ("" + int) !== value) { json[name] = value; } else { json[name] = int; } } _ref1 = node.children; for (_i = 0, _len = _ref1.length; _i < _len; _i++) { n = _ref1[_i]; name = n.name; if (n.getAttribute("isArray") === "true") { json[name] = parse_array(n); } else { json[name] = parse_object(n); } } return json; }; return parse_object(m); }, encodeMessageToXml: function(m, json) { var encode_array, encode_object; encode_object = function(m, json) { var name, value; for (name in json) { value = json[name]; if (value == null) { } else if (value.constructor === Object) { encode_object(m.c(name), value); } else if (value.constructor === Array) { encode_array(m.c(name), value); } else { m.setAttribute(name, value); } } return m; }; encode_array = function(m, array) { var e, _i, _len; m.setAttribute("isArray", "true"); for (_i = 0, _len = array.length; _i < _len; _i++) { e = array[_i]; if (e.constructor === Object) { encode_object(m.c("array-element"), e); } else { encode_array(m.c("array-element"), e); } } return m; }; if (json.constructor === Object) { return encode_object(m.c("y", { xmlns: "http://y.ninja/connector-stanza" }), json); } else if (json.constructor === Array) { return encode_array(m.c("y", { xmlns: "http://y.ninja/connector-stanza" }), json); } else { throw new Error("I can't encode this json!"); } }, setIsBoundToY: function() { if (typeof this.on_bound_to_y === "function") { this.on_bound_to_y(); } delete this.when_bound_to_y; return this.is_bound_to_y = true; } };