[tor-commits] [flashproxy/rtmfp] Add an RTMFPSocket class.
dcf at torproject.org
dcf at torproject.org
Sun Jun 12 08:56:32 UTC 2011
commit 9a85e5b90a37c8c989cda04f69862b82f25ca924
Author: David Fifield <david at bamsoftware.com>
Date: Sat Jun 11 21:26:55 2011 -0700
Add an RTMFPSocket class.
---
RTMFPSocket.as | 209 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++
1 files changed, 209 insertions(+), 0 deletions(-)
diff --git a/RTMFPSocket.as b/RTMFPSocket.as
new file mode 100644
index 0000000..e23c4fd
--- /dev/null
+++ b/RTMFPSocket.as
@@ -0,0 +1,209 @@
+/*
+The RTMFPSocket class provides a socket-like interface around RTMFP
+NetConnection and NetStream. Each RTMFPSocket contains one NetConnection and two
+NetStreams, one for reading and one for writing.
+
+To create a listening socket:
+ var rs:RTMFPSocket = new RTMFPSocket(url, key);
+ rs.addEventListener(Event.COMPLETE, function (e:Event):void {
+ // rs.id is set and can be sent out of band to the client.
+ });
+ rs.addEventListener(RTMFPSocket.ACCEPT_EVENT, function (e:Event):void {
+ // rs.peer_id is the ID of the connected client.
+ });
+ rs.listen();
+To connect to a listening socket:
+ // Receive peer_id out of band.
+ var rs:RTMFPSocket = new RTMFPSocket(url, key);
+ rs.addEventListener(Event.CONNECT, function (e:Event):void {
+ // rs.id and rs.peer_id are now set.
+ });
+*/
+
+package
+{
+ import flash.events.Event;
+ import flash.events.EventDispatcher;
+ import flash.events.IOErrorEvent;
+ import flash.events.NetStatusEvent;
+ import flash.events.ProgressEvent;
+ import flash.net.NetConnection;
+ import flash.net.NetStream;
+ import flash.utils.ByteArray;
+
+ public class RTMFPSocket extends EventDispatcher
+ {
+ public static const ACCEPT_EVENT:String = "accept";
+
+ public var connected:Boolean;
+
+ private var nc:NetConnection;
+ private var incoming:NetStream;
+ private var outgoing:NetStream;
+
+ /* Cache to hold the peer ID between when connect is called and the
+ NetConnection exists. */
+ private var connect_peer_id:String;
+
+ private var buffer:ByteArray;
+
+ private var cirrus_url:String;
+ private var cirrus_key:String;
+
+ public function RTMFPSocket(cirrus_url:String, cirrus_key:String)
+ {
+ connected = false;
+
+ buffer = new ByteArray();
+
+ this.cirrus_url = cirrus_url;
+ this.cirrus_key = cirrus_key;
+
+ nc = new NetConnection();
+ }
+
+ public function get id():String
+ {
+ return nc.nearID;
+ }
+
+ public function get peer_id():String
+ {
+ return incoming.farID;
+ }
+
+ /* NetStatusEvents that aren't handled more specifically in
+ listen_netstatus_event or connect_netstatus_event. */
+ private function generic_netstatus_event(e:NetStatusEvent):void
+ {
+ switch (e.info.code) {
+ case "NetConnection.Connect.Closed":
+ dispatchEvent(new Event(Event.CLOSE));
+ break;
+ case "NetStream.Connect.Closed":
+ connected = false;
+ close();
+ break;
+ default:
+ var event:IOErrorEvent = new IOErrorEvent(IOErrorEvent.IO_ERROR);
+ event.text = e.info.code;
+ dispatchEvent(event);
+ break;
+ }
+ }
+
+ private function listen_netstatus_event(e:NetStatusEvent):void
+ {
+ switch (e.info.code) {
+ case "NetConnection.Connect.Success":
+ outgoing = new NetStream(nc, NetStream.DIRECT_CONNECTIONS);
+ outgoing.client = {
+ onPeerConnect: listen_onpeerconnect
+ };
+ outgoing.publish("server");
+
+ /* listen is complete, ready to accept. */
+ dispatchEvent(new Event(Event.COMPLETE));
+ break;
+ case "NetStream.Connect.Success":
+ break;
+ default:
+ return generic_netstatus_event(e);
+ break;
+ }
+ }
+
+ private function listen_onpeerconnect(peer:NetStream):Boolean {
+ incoming = new NetStream(nc, peer.farID);
+ incoming.client = {
+ r: receive_data
+ };
+ incoming.play("client");
+
+ connected = true;
+ dispatchEvent(new Event(ACCEPT_EVENT));
+
+ return true;
+ }
+
+ private function connect_netstatus_event(e:NetStatusEvent):void
+ {
+ switch (e.info.code) {
+ case "NetConnection.Connect.Success":
+ outgoing = new NetStream(nc, NetStream.DIRECT_CONNECTIONS);
+ outgoing.publish("client");
+
+ incoming = new NetStream(nc, connect_peer_id);
+ incoming.client = {
+ r: receive_data
+ };
+ incoming.play("server");
+ break;
+ case "NetStream.Connect.Success":
+ connected = true;
+ dispatchEvent(new Event(Event.CONNECT));
+ break;
+ default:
+ return generic_netstatus_event(e);
+ break;
+ }
+ }
+
+ /* Function called back when the other side does a send. */
+ private function receive_data(bytes:ByteArray):void {
+ var event:ProgressEvent;
+
+ event = new ProgressEvent(ProgressEvent.SOCKET_DATA);
+ event.bytesLoaded = bytes.bytesAvailable;
+
+ bytes.readBytes(buffer, buffer.length, bytes.bytesAvailable);
+ if (bytes.bytesAvailable == 0) {
+ /* Reclaim memory space. */
+ bytes.position = 0;
+ bytes.length = 0;
+ }
+
+ dispatchEvent(event);
+ }
+
+ public function listen():void
+ {
+ nc.addEventListener(NetStatusEvent.NET_STATUS, listen_netstatus_event);
+ nc.connect(cirrus_url, cirrus_key);
+ }
+
+ public function connect(peer_id:String):void
+ {
+ /* Store for later reading by connect_netstatus_event. */
+ this.connect_peer_id = peer_id;
+
+ nc.addEventListener(NetStatusEvent.NET_STATUS, connect_netstatus_event);
+ nc.connect(cirrus_url, cirrus_key);
+ }
+
+ public function close():void
+ {
+ outgoing.close();
+ incoming.close();
+ nc.close();
+ }
+
+ public function readBytes(output:ByteArray, offset:uint = 0, length:uint = 0):void
+ {
+ buffer.readBytes(output, offset, length);
+ }
+
+ public function writeBytes(input:ByteArray, offset:uint = 0, length:uint = 0):void
+ {
+ var sendbuf:ByteArray;
+
+ /* Read into a new buffer, in case offset and length do not
+ completely span input. */
+ sendbuf = new ByteArray();
+ sendbuf.writeBytes(input, offset, length);
+
+ /* Use a short method name because it's sent over the wire. */
+ outgoing.send("r", sendbuf);
+ }
+ }
+}
More information about the tor-commits
mailing list