212 lines
23 KiB
JavaScript
212 lines
23 KiB
JavaScript
(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
|
|
var Connector;
|
|
|
|
Connector = (function() {
|
|
function Connector() {
|
|
this.is_synced = false;
|
|
this.compute_when_synced = [];
|
|
this.connections = {};
|
|
this.unsynced_connections = {};
|
|
this.receive_handlers = [];
|
|
this.sync_process_order = [];
|
|
}
|
|
|
|
Connector.prototype.whenSynced = function(args) {
|
|
if (this.is_synced) {
|
|
return args[0].apply(this, args.slice(1));
|
|
} else {
|
|
return this.compute_when_synced.push(args);
|
|
}
|
|
};
|
|
|
|
Connector.prototype.whenReceiving = function(f) {
|
|
return this.receive_handlers.push(f);
|
|
};
|
|
|
|
Connector.prototype.multicast = function(peers, message) {
|
|
return this.whenSynced([_send, peers, message]);
|
|
};
|
|
|
|
Connector.prototype.unicast = function(peer, message) {
|
|
return this.whenSynced([_send, peer, message]);
|
|
};
|
|
|
|
Connector.prototype.broadcast = function(message) {
|
|
return this.whenSynced([
|
|
(function(_this) {
|
|
return function() {
|
|
var peer, peerid, _ref, _results;
|
|
_ref = _this.connections;
|
|
_results = [];
|
|
for (peerid in _ref) {
|
|
peer = _ref[peerid];
|
|
_results.push(_this._send(peerid, message));
|
|
}
|
|
return _results;
|
|
};
|
|
})(this)
|
|
]);
|
|
};
|
|
|
|
Connector.prototype.whenSyncing = function() {
|
|
var i, _i, _ref, _results;
|
|
_results = [];
|
|
for (i = _i = _ref = arguments.length - 1; _ref <= 0 ? _i <= 0 : _i >= 0; i = _ref <= 0 ? ++_i : --_i) {
|
|
_results.push(this.sync_process_order.unshift(arguments[i]));
|
|
}
|
|
return _results;
|
|
};
|
|
|
|
return Connector;
|
|
|
|
})();
|
|
|
|
module.exports = Connector;
|
|
|
|
|
|
|
|
},{}],2:[function(require,module,exports){
|
|
var Connector, PeerJsConnector,
|
|
__bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; },
|
|
__hasProp = {}.hasOwnProperty,
|
|
__extends = function(child, parent) { for (var key in parent) { if (__hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; };
|
|
|
|
Connector = require('../connector');
|
|
|
|
window.PeerJsConnector = PeerJsConnector = (function(_super) {
|
|
__extends(PeerJsConnector, _super);
|
|
|
|
function PeerJsConnector(id, options) {
|
|
var that;
|
|
this.id = id;
|
|
this._addConnection = __bind(this._addConnection, this);
|
|
PeerJsConnector.__super__.constructor.call(this);
|
|
that = this;
|
|
this.sync_process_order.push(function() {
|
|
var conn, peerid, peers;
|
|
peers = (function() {
|
|
var _ref, _results;
|
|
_ref = that.connections;
|
|
_results = [];
|
|
for (peerid in _ref) {
|
|
conn = _ref[peerid];
|
|
_results.push(peerid);
|
|
}
|
|
return _results;
|
|
})();
|
|
return peers;
|
|
});
|
|
this.sync_process_order.push(function(peers) {
|
|
var peerid, _i, _len;
|
|
for (_i = 0, _len = peers.length; _i < _len; _i++) {
|
|
peerid = peers[_i];
|
|
that.join(peerid);
|
|
}
|
|
return true;
|
|
});
|
|
this.conn = new Peer(this.id, options);
|
|
this.conn.on('error', function(err) {
|
|
throw new Error("Peerjs connector: " + err);
|
|
});
|
|
this.conn.on('disconnected', function() {
|
|
throw new Error("Peerjs connector disconnected from signalling server. Cannot accept new connections. Not fatal, but not so good either..");
|
|
});
|
|
this.conn.on('disconnect', function() {
|
|
return that.conn.reconnect();
|
|
});
|
|
this.conn.on('connection', this._addConnection);
|
|
}
|
|
|
|
PeerJsConnector.prototype.join = function(peerid) {
|
|
var peer;
|
|
if ((this.unsynced_connections[peerid] == null) && (this.connections[peerid] == null) && peerid !== this.id) {
|
|
peer = this.conn.connect(peerid, {
|
|
reliable: true
|
|
});
|
|
this.unsynced_connections[peerid] = peer;
|
|
this._addConnection(peer);
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
};
|
|
|
|
PeerJsConnector.prototype._send = function(peer_s, message) {
|
|
var error, errors, peer, _i, _len;
|
|
if (peer_s.constructor === [].constructor) {
|
|
errors = [];
|
|
for (_i = 0, _len = peer_s.length; _i < _len; _i++) {
|
|
peer = peer_s[_i];
|
|
try {
|
|
this.connection[peer].send(message);
|
|
} catch (_error) {
|
|
error = _error;
|
|
errors.push(error + "");
|
|
}
|
|
}
|
|
if (errors.length > 0) {
|
|
throw new Error(errors);
|
|
}
|
|
} else {
|
|
return this.connections[peer_s].send(message);
|
|
}
|
|
};
|
|
|
|
PeerJsConnector.prototype._addConnection = function(peer) {
|
|
return peer.on('open', (function(_this) {
|
|
return function() {
|
|
var current_sync_i, that;
|
|
that = _this;
|
|
peer.send(that.sync_process_order[0]());
|
|
current_sync_i = 1;
|
|
return peer.on('data', function(data) {
|
|
var comp, f, isEmpty, _i, _j, _len, _len1, _ref, _ref1, _results;
|
|
console.log("receive data: " + (JSON.stringify(data)));
|
|
if (current_sync_i < that.sync_process_order.length) {
|
|
return peer.send(that.sync_process_order[current_sync_i++].call(that, data));
|
|
} else if (current_sync_i === that.sync_process_order.length) {
|
|
current_sync_i++;
|
|
delete that.unsynced_connections[peer.peer];
|
|
that.connections[peer.peer] = peer;
|
|
peer.on('close', function() {
|
|
return delete that.connections[peer.peer];
|
|
});
|
|
isEmpty = function(os) {
|
|
var o;
|
|
for (o in os) {
|
|
return false;
|
|
}
|
|
return true;
|
|
};
|
|
if (isEmpty(that.unsynced_connections)) {
|
|
that.is_synced = true;
|
|
_ref = that.compute_when_synced;
|
|
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
|
comp = _ref[_i];
|
|
comp[0].apply(that, comp.slice(1));
|
|
}
|
|
return that.compute_when_synced = [];
|
|
}
|
|
} else {
|
|
_ref1 = that.receive_handlers;
|
|
_results = [];
|
|
for (_j = 0, _len1 = _ref1.length; _j < _len1; _j++) {
|
|
f = _ref1[_j];
|
|
_results.push(f(peer.peer, data));
|
|
}
|
|
return _results;
|
|
}
|
|
});
|
|
};
|
|
})(this));
|
|
};
|
|
|
|
return PeerJsConnector;
|
|
|
|
})(Connector);
|
|
|
|
|
|
|
|
},{"../connector":1}]},{},[2])
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/home/codio/workspace/node_modules/gulp-browserify/node_modules/browserify/node_modules/browser-pack/_prelude.js","/home/codio/workspace/lib/connector.coffee","/home/codio/workspace/lib/peerjs-connector/peerjs-connector.coffee"],"names":[],"mappings":"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","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error(\"Cannot find module '\"+o+\"'\")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","\nclass Connector\n  \n  constructor: ()->\n    # is set to true when this is synced with all other connections\n    @is_synced = false\n    # compute all of these functions when all connections are synced.\n    @compute_when_synced = []\n    # Peerjs Connections: key: conn-id, value: conn\n    @connections = {}\n    # Connections, that have been initialized, but have not been (fully) synced yet.\n    @unsynced_connections = {}\n    # List of functions that shall process incoming data\n    @receive_handlers = []\n    # A list of functions that are executed (left to right) when syncing with a peer. \n    @sync_process_order = []\n    \n  #\n  # Execute a function _when_ we are connected. If not connected, wait until connected.\n  # @param f {Function} Will be executed on the PeerJs-Connector context.\n  #\n  whenSynced: (args)->\n    if @is_synced\n      args[0].apply this, args[1..]\n    else\n      @compute_when_synced.push args \n  \n  #\n  # Execute an function _when_ a message is received.\n  # @param f {Function} Will be executed on the PeerJs-Connector context. f will be called with (sender_id, broadcast {true|false}, message).\n  #\n  whenReceiving: (f)->\n    @receive_handlers.push f\n  \n  #\n  # Send a message to a (sub)-set of all connected peers.\n  # @param peers {Array<connection_ids>} A set of ids.\n  # @param message {Object} The message to send.\n  #\n  multicast: (peers, message)->\n    @whenSynced [_send, peers, message]\n  \n  #\n  # Send a message to one of the connected peers.\n  # @param peers {connection_id} A connection id.\n  # @param message {Object} The message to send.\n  #\n  unicast: (peer, message)->\n    @whenSynced [_send, peer, message]\n  \n  # \n  # Broadcast a message to all connected peers.\n  # @param message {Object} The message to broadcast.\n  # \n  broadcast: (message)->\n    @whenSynced [()=>\n      for peerid,peer of @connections\n        @_send peerid, message]\n \n  #\n  # Define how you want to handle the sync process of two users.\n  # This is a synchronous handshake. Every user will perform exactly the same actions at the same time. E.g.\n  # @example\n  #   whenSyncing(function(){ // first call must not have parameters!\n  #       return this.id; // Send the id of this connector.\n  #   },function(peerid){ // you receive the peerid of the other connections.\n  #       // you can do something with the peerid\n  #       // return \"you are my friend\"; // you could send another massage.\n  #   }); // this is the end of the sync process.\n  #\n  whenSyncing: ()->\n    for i in [(arguments.length-1)..0]\n      @sync_process_order.unshift arguments[i]\n\n\n\nmodule.exports = Connector\n","Connector = require '../connector'\n      \nwindow.PeerJsConnector = class PeerJsConnector extends Connector\n  \n  constructor: (@id, options)->\n    super()\n    that = this\n    # The following two functions should be performed at the end of the syncing process.\n    # In peerjs all connection ids must be send. \n    @sync_process_order.push ()->\n      peers = for peerid,conn of that.connections \n        peerid\n      peers \n    # Then connect to the connection ids. \n    @sync_process_order.push (peers)->\n      for peerid in peers \n          that.join peerid\n      true \n    # Create the Peerjs instance\n    @conn = new Peer @id, options\n    # TODO: improve error handling, what happens if disconnected? provide feedback\n    @conn.on 'error', (err)->\n      throw new Error \"Peerjs connector: #{err}\"\n    @conn.on 'disconnected', ()->\n      throw new Error \"Peerjs connector disconnected from signalling server. Cannot accept new connections. Not fatal, but not so good either..\"\n    @conn.on 'disconnect', ()->\n      that.conn.reconnect()\n    @conn.on 'connection', @_addConnection\n  \n  #\n  # Join a communication room. In case of peerjs, you just have to join to one other client. This connector will join to the other peers automatically.\n  # @param id {String} The connection id of another client.\n  #\n  join: (peerid)->\n    if not @unsynced_connections[peerid]? and not @connections[peerid]? and peerid isnt @id\n      peer = @conn.connect peerid, {reliable: true} \n      @unsynced_connections[peerid] = peer\n      @_addConnection peer\n      true\n    else\n      false\n  \n  #\n  # Send a message to a peer or set of peers. This is peerjs specific.\n  # @overload _send(peerid, message)\n  #   @param peerid {String} PeerJs connection id of _another_ peer\n  #   @param message {Object} Some object that shall be send\n  # @overload _send(peerids, message)\n  #   @param peerids {Array<String>} PeerJs connection ids of _other_ peers\n  #   @param message {Object} Some object that shall be send\n  #\n  _send: (peer_s, message)->\n    if peer_s.constructor is [].constructor\n      # Throw errors _after_ the message has been send to all other peers. \n      # Just in case a connection is invalid.\n      errors = []\n      for peer in peer_s\n        try\n          @connection[peer].send message\n        catch error \n          errors.push(error+\"\")\n      if errors.length > 0\n        throw new Error errors \n    else\n      @connections[peer_s].send message\n    \n  #\n  # @private\n  # This is a helper function that is only related to the peerjs connector. \n  # Connect to another peer.\n  _addConnection: (peer)=>\n    peer.on 'open', ()=>\n      that = @\n      peer.send that.sync_process_order[0]()\n      current_sync_i = 1\n      peer.on 'data', (data)->\n        console.log(\"receive data: #{JSON.stringify data}\")\n        if current_sync_i < that.sync_process_order.length\n          peer.send that.sync_process_order[current_sync_i++].call that, data\n        else if current_sync_i is that.sync_process_order.length\n          # All sync functions have been called. Increment current_sync_i one last time\n          current_sync_i++\n          # add it to the connections object\n          delete that.unsynced_connections[peer.peer]\n          that.connections[peer.peer] = peer\n          # when the conn closes, delete it from the connections object\n          peer.on 'close', ()->\n            delete that.connections[peer.peer]\n          # helper fkt. true iff os is an object that does not hold enumerable properties\n          isEmpty = (os)->\n            for o of os\n              return false\n            return true\n          if isEmpty(that.unsynced_connections)\n            # there are no unsynced connections. we are now synced. \n            # therefore execute all fkts in this.compute_when_synced\n            that.is_synced = true\n            for comp in that.compute_when_synced\n              comp[0].apply that, comp[1..]\n            that.compute_when_synced = []\n        else\n          # you received a new message, that is not a sync message.\n          # notify the receive_handlers\n          for f in that.receive_handlers \n            f peer.peer, data\n\n\n      "]}
|