Aphlict, simple notification server
Summary:
This is purely a prototype at the moment, but the basic functionality sort of
works.
I'm not sure how far I want to go with this but I think we might be able to get
somewhere without it being gross.
The idea here is to build a notification server WITHOUT using Comet, since Comet
is extremely difficult and complicated.
Instead, I use Flash on the client. LocalConnection allows flash instances to
talk to each other and connect() can be used as a locking primitive. This allows
all the instances to elect a master instance in a race-safe way. The master is
responsible for opening a single connnection to the server.
On the server, I use Node.js since PHP is pretty unsuitable for this task.
See Github Issue #3: https://github.com/facebook/phabricator/issues/3
One thing I need to figure out next is if I can reasonably do SSL/TSL over Flash
(it looks like I can, in theory, with the as3crypto library) or if the server
needs to just send down version information and trigger a separate Ajax call on
the client.
Test Plan:
Created a client pool and connected it to the server, with election and failover
apparently working correctly.
Reviewed By: aran
Reviewers: Girish, aran, jungejason, tuomaspelkonen, davidrecordon
Commenters: Girish, davidrecordon
CC: aran, epriestley, Girish, davidrecordon
Differential Revision: 284
2011-05-15 23:05:02 +02:00
|
|
|
package {
|
|
|
|
|
|
|
|
import flash.net.*;
|
|
|
|
import flash.utils.*;
|
|
|
|
import flash.media.*;
|
|
|
|
import flash.display.*;
|
|
|
|
import flash.events.*;
|
|
|
|
import flash.external.ExternalInterface;
|
|
|
|
|
|
|
|
import com.phabricator.*;
|
|
|
|
|
|
|
|
import vegas.strings.JSON;
|
|
|
|
|
|
|
|
public class Aphlict extends Sprite {
|
|
|
|
|
|
|
|
private var client:String;
|
|
|
|
|
|
|
|
private var master:LocalConnection;
|
|
|
|
private var recv:LocalConnection;
|
|
|
|
private var send:LocalConnection;
|
|
|
|
|
|
|
|
private var receiver:AphlictReceiver;
|
|
|
|
private var loyalUntil:Number = 0;
|
|
|
|
private var subjects:Array;
|
|
|
|
private var frequency:Number = 100;
|
|
|
|
|
|
|
|
private var socket:Socket;
|
|
|
|
private var readBuffer:ByteArray;
|
|
|
|
|
2012-03-07 05:14:03 +01:00
|
|
|
private var remoteServer:String;
|
|
|
|
private var remotePort:Number;
|
|
|
|
|
Aphlict, simple notification server
Summary:
This is purely a prototype at the moment, but the basic functionality sort of
works.
I'm not sure how far I want to go with this but I think we might be able to get
somewhere without it being gross.
The idea here is to build a notification server WITHOUT using Comet, since Comet
is extremely difficult and complicated.
Instead, I use Flash on the client. LocalConnection allows flash instances to
talk to each other and connect() can be used as a locking primitive. This allows
all the instances to elect a master instance in a race-safe way. The master is
responsible for opening a single connnection to the server.
On the server, I use Node.js since PHP is pretty unsuitable for this task.
See Github Issue #3: https://github.com/facebook/phabricator/issues/3
One thing I need to figure out next is if I can reasonably do SSL/TSL over Flash
(it looks like I can, in theory, with the as3crypto library) or if the server
needs to just send down version information and trigger a separate Ajax call on
the client.
Test Plan:
Created a client pool and connected it to the server, with election and failover
apparently working correctly.
Reviewed By: aran
Reviewers: Girish, aran, jungejason, tuomaspelkonen, davidrecordon
Commenters: Girish, davidrecordon
CC: aran, epriestley, Girish, davidrecordon
Differential Revision: 284
2011-05-15 23:05:02 +02:00
|
|
|
public function Aphlict() {
|
|
|
|
super();
|
|
|
|
|
2012-03-07 05:14:03 +01:00
|
|
|
ExternalInterface.addCallback('connect', this.externalConnect);
|
|
|
|
ExternalInterface.call(
|
|
|
|
'JX.Stratcom.invoke',
|
|
|
|
'aphlict-component-ready',
|
|
|
|
null,
|
|
|
|
{});
|
|
|
|
}
|
|
|
|
|
|
|
|
public function externalConnect(server:String, port:Number):void {
|
|
|
|
this.externalInvoke('connect');
|
|
|
|
|
|
|
|
this.remoteServer = server;
|
|
|
|
this.remotePort = port;
|
|
|
|
|
Aphlict, simple notification server
Summary:
This is purely a prototype at the moment, but the basic functionality sort of
works.
I'm not sure how far I want to go with this but I think we might be able to get
somewhere without it being gross.
The idea here is to build a notification server WITHOUT using Comet, since Comet
is extremely difficult and complicated.
Instead, I use Flash on the client. LocalConnection allows flash instances to
talk to each other and connect() can be used as a locking primitive. This allows
all the instances to elect a master instance in a race-safe way. The master is
responsible for opening a single connnection to the server.
On the server, I use Node.js since PHP is pretty unsuitable for this task.
See Github Issue #3: https://github.com/facebook/phabricator/issues/3
One thing I need to figure out next is if I can reasonably do SSL/TSL over Flash
(it looks like I can, in theory, with the as3crypto library) or if the server
needs to just send down version information and trigger a separate Ajax call on
the client.
Test Plan:
Created a client pool and connected it to the server, with election and failover
apparently working correctly.
Reviewed By: aran
Reviewers: Girish, aran, jungejason, tuomaspelkonen, davidrecordon
Commenters: Girish, davidrecordon
CC: aran, epriestley, Girish, davidrecordon
Differential Revision: 284
2011-05-15 23:05:02 +02:00
|
|
|
this.master = null;
|
|
|
|
this.receiver = new AphlictReceiver(this);
|
|
|
|
this.subjects = [];
|
|
|
|
|
|
|
|
this.send = new LocalConnection();
|
|
|
|
|
|
|
|
this.recv = new LocalConnection();
|
|
|
|
this.recv.client = this.receiver;
|
|
|
|
for (var ii:Number = 0; ii < 32; ii++) {
|
|
|
|
try {
|
|
|
|
this.recv.connect('aphlict_subject_' + ii);
|
|
|
|
this.client = 'aphlict_subject_' + ii;
|
|
|
|
} catch (x:Error) {
|
|
|
|
// Some other Aphlict client is holding that ID.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!this.client) {
|
|
|
|
// Too many clients open already, just exit.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.usurp();
|
|
|
|
}
|
|
|
|
|
|
|
|
private function usurp():void {
|
|
|
|
if (this.master) {
|
|
|
|
for (var ii:Number = 0; ii < this.subjects.length; ii++) {
|
|
|
|
if (this.subjects[ii] == this.client) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
this.send.send(this.subjects[ii], 'remainLoyal');
|
|
|
|
}
|
|
|
|
} else if (this.loyalUntil < new Date().getTime()) {
|
|
|
|
var recv:LocalConnection = new LocalConnection();
|
|
|
|
recv.client = this.receiver;
|
|
|
|
try {
|
|
|
|
recv.connect('aphlict_master');
|
|
|
|
this.master = recv;
|
|
|
|
this.subjects = [this.client];
|
|
|
|
|
|
|
|
this.connectToServer();
|
|
|
|
|
|
|
|
} catch (x:Error) {
|
|
|
|
// Can't become the master.
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!this.master) {
|
|
|
|
this.send.send('aphlict_master', 'becomeLoyal', this.client);
|
|
|
|
this.remainLoyal();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
setTimeout(this.usurp, this.frequency);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function connectToServer():void {
|
|
|
|
var socket:Socket = new Socket();
|
|
|
|
|
|
|
|
socket.addEventListener(Event.CONNECT, didConnectSocket);
|
|
|
|
socket.addEventListener(Event.CLOSE, didCloseSocket);
|
|
|
|
socket.addEventListener(ProgressEvent.SOCKET_DATA, didReceiveSocket);
|
|
|
|
|
2012-03-07 05:14:03 +01:00
|
|
|
socket.addEventListener(IOErrorEvent.IO_ERROR, didIOErrorSocket);
|
|
|
|
socket.addEventListener(
|
|
|
|
SecurityErrorEvent.SECURITY_ERROR,
|
|
|
|
didSecurityErrorSocket);
|
|
|
|
|
|
|
|
socket.connect(this.remoteServer, this.remotePort);
|
Aphlict, simple notification server
Summary:
This is purely a prototype at the moment, but the basic functionality sort of
works.
I'm not sure how far I want to go with this but I think we might be able to get
somewhere without it being gross.
The idea here is to build a notification server WITHOUT using Comet, since Comet
is extremely difficult and complicated.
Instead, I use Flash on the client. LocalConnection allows flash instances to
talk to each other and connect() can be used as a locking primitive. This allows
all the instances to elect a master instance in a race-safe way. The master is
responsible for opening a single connnection to the server.
On the server, I use Node.js since PHP is pretty unsuitable for this task.
See Github Issue #3: https://github.com/facebook/phabricator/issues/3
One thing I need to figure out next is if I can reasonably do SSL/TSL over Flash
(it looks like I can, in theory, with the as3crypto library) or if the server
needs to just send down version information and trigger a separate Ajax call on
the client.
Test Plan:
Created a client pool and connected it to the server, with election and failover
apparently working correctly.
Reviewed By: aran
Reviewers: Girish, aran, jungejason, tuomaspelkonen, davidrecordon
Commenters: Girish, davidrecordon
CC: aran, epriestley, Girish, davidrecordon
Differential Revision: 284
2011-05-15 23:05:02 +02:00
|
|
|
|
|
|
|
this.readBuffer = new ByteArray();
|
|
|
|
this.socket = socket;
|
|
|
|
}
|
|
|
|
|
|
|
|
private function didConnectSocket(event:Event):void {
|
2012-03-07 05:14:03 +01:00
|
|
|
this.externalInvoke('connected');
|
Aphlict, simple notification server
Summary:
This is purely a prototype at the moment, but the basic functionality sort of
works.
I'm not sure how far I want to go with this but I think we might be able to get
somewhere without it being gross.
The idea here is to build a notification server WITHOUT using Comet, since Comet
is extremely difficult and complicated.
Instead, I use Flash on the client. LocalConnection allows flash instances to
talk to each other and connect() can be used as a locking primitive. This allows
all the instances to elect a master instance in a race-safe way. The master is
responsible for opening a single connnection to the server.
On the server, I use Node.js since PHP is pretty unsuitable for this task.
See Github Issue #3: https://github.com/facebook/phabricator/issues/3
One thing I need to figure out next is if I can reasonably do SSL/TSL over Flash
(it looks like I can, in theory, with the as3crypto library) or if the server
needs to just send down version information and trigger a separate Ajax call on
the client.
Test Plan:
Created a client pool and connected it to the server, with election and failover
apparently working correctly.
Reviewed By: aran
Reviewers: Girish, aran, jungejason, tuomaspelkonen, davidrecordon
Commenters: Girish, davidrecordon
CC: aran, epriestley, Girish, davidrecordon
Differential Revision: 284
2011-05-15 23:05:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private function didCloseSocket(event:Event):void {
|
2012-03-07 05:14:03 +01:00
|
|
|
this.externalInvoke('close');
|
|
|
|
}
|
|
|
|
|
|
|
|
private function didIOErrorSocket(event:IOErrorEvent):void {
|
|
|
|
this.externalInvoke('error', event.text);
|
Aphlict, simple notification server
Summary:
This is purely a prototype at the moment, but the basic functionality sort of
works.
I'm not sure how far I want to go with this but I think we might be able to get
somewhere without it being gross.
The idea here is to build a notification server WITHOUT using Comet, since Comet
is extremely difficult and complicated.
Instead, I use Flash on the client. LocalConnection allows flash instances to
talk to each other and connect() can be used as a locking primitive. This allows
all the instances to elect a master instance in a race-safe way. The master is
responsible for opening a single connnection to the server.
On the server, I use Node.js since PHP is pretty unsuitable for this task.
See Github Issue #3: https://github.com/facebook/phabricator/issues/3
One thing I need to figure out next is if I can reasonably do SSL/TSL over Flash
(it looks like I can, in theory, with the as3crypto library) or if the server
needs to just send down version information and trigger a separate Ajax call on
the client.
Test Plan:
Created a client pool and connected it to the server, with election and failover
apparently working correctly.
Reviewed By: aran
Reviewers: Girish, aran, jungejason, tuomaspelkonen, davidrecordon
Commenters: Girish, davidrecordon
CC: aran, epriestley, Girish, davidrecordon
Differential Revision: 284
2011-05-15 23:05:02 +02:00
|
|
|
}
|
|
|
|
|
2012-03-07 05:14:03 +01:00
|
|
|
private function didSecurityErrorSocket(event:SecurityErrorEvent):void {
|
|
|
|
this.externalInvoke('error', event.text);
|
Aphlict, simple notification server
Summary:
This is purely a prototype at the moment, but the basic functionality sort of
works.
I'm not sure how far I want to go with this but I think we might be able to get
somewhere without it being gross.
The idea here is to build a notification server WITHOUT using Comet, since Comet
is extremely difficult and complicated.
Instead, I use Flash on the client. LocalConnection allows flash instances to
talk to each other and connect() can be used as a locking primitive. This allows
all the instances to elect a master instance in a race-safe way. The master is
responsible for opening a single connnection to the server.
On the server, I use Node.js since PHP is pretty unsuitable for this task.
See Github Issue #3: https://github.com/facebook/phabricator/issues/3
One thing I need to figure out next is if I can reasonably do SSL/TSL over Flash
(it looks like I can, in theory, with the as3crypto library) or if the server
needs to just send down version information and trigger a separate Ajax call on
the client.
Test Plan:
Created a client pool and connected it to the server, with election and failover
apparently working correctly.
Reviewed By: aran
Reviewers: Girish, aran, jungejason, tuomaspelkonen, davidrecordon
Commenters: Girish, davidrecordon
CC: aran, epriestley, Girish, davidrecordon
Differential Revision: 284
2011-05-15 23:05:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private function didReceiveSocket(event:Event):void {
|
|
|
|
var b:ByteArray = this.readBuffer;
|
|
|
|
this.socket.readBytes(b, b.length);
|
|
|
|
|
|
|
|
do {
|
|
|
|
b = this.readBuffer;
|
|
|
|
b.position = 0;
|
|
|
|
|
|
|
|
if (b.length <= 8) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
var msg_len:Number = parseInt(b.readUTFBytes(8), 10);
|
|
|
|
if (b.length >= msg_len + 8) {
|
|
|
|
var bytes:String = b.readUTFBytes(msg_len);
|
2012-03-07 05:14:03 +01:00
|
|
|
var data:Object = vegas.strings.JSON.deserialize(bytes);
|
Aphlict, simple notification server
Summary:
This is purely a prototype at the moment, but the basic functionality sort of
works.
I'm not sure how far I want to go with this but I think we might be able to get
somewhere without it being gross.
The idea here is to build a notification server WITHOUT using Comet, since Comet
is extremely difficult and complicated.
Instead, I use Flash on the client. LocalConnection allows flash instances to
talk to each other and connect() can be used as a locking primitive. This allows
all the instances to elect a master instance in a race-safe way. The master is
responsible for opening a single connnection to the server.
On the server, I use Node.js since PHP is pretty unsuitable for this task.
See Github Issue #3: https://github.com/facebook/phabricator/issues/3
One thing I need to figure out next is if I can reasonably do SSL/TSL over Flash
(it looks like I can, in theory, with the as3crypto library) or if the server
needs to just send down version information and trigger a separate Ajax call on
the client.
Test Plan:
Created a client pool and connected it to the server, with election and failover
apparently working correctly.
Reviewed By: aran
Reviewers: Girish, aran, jungejason, tuomaspelkonen, davidrecordon
Commenters: Girish, davidrecordon
CC: aran, epriestley, Girish, davidrecordon
Differential Revision: 284
2011-05-15 23:05:02 +02:00
|
|
|
var t:ByteArray = new ByteArray();
|
|
|
|
t.writeBytes(b, msg_len + 8);
|
|
|
|
this.readBuffer = t;
|
|
|
|
|
|
|
|
for (var ii:Number = 0; ii < this.subjects.length; ii++) {
|
|
|
|
this.send.send(this.subjects[ii], 'receiveMessage', data);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} while (true);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
public function remainLoyal():void {
|
|
|
|
this.loyalUntil = new Date().getTime() + (2 * this.frequency);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function becomeLoyal(subject:String):void {
|
|
|
|
this.subjects.push(subject);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function receiveMessage(msg:Object):void {
|
2012-03-07 05:14:03 +01:00
|
|
|
this.externalInvoke('receive', msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function externalInvoke(type:String, object:Object = null):void {
|
|
|
|
ExternalInterface.call('JX.Aphlict.didReceiveEvent', type, object);
|
Aphlict, simple notification server
Summary:
This is purely a prototype at the moment, but the basic functionality sort of
works.
I'm not sure how far I want to go with this but I think we might be able to get
somewhere without it being gross.
The idea here is to build a notification server WITHOUT using Comet, since Comet
is extremely difficult and complicated.
Instead, I use Flash on the client. LocalConnection allows flash instances to
talk to each other and connect() can be used as a locking primitive. This allows
all the instances to elect a master instance in a race-safe way. The master is
responsible for opening a single connnection to the server.
On the server, I use Node.js since PHP is pretty unsuitable for this task.
See Github Issue #3: https://github.com/facebook/phabricator/issues/3
One thing I need to figure out next is if I can reasonably do SSL/TSL over Flash
(it looks like I can, in theory, with the as3crypto library) or if the server
needs to just send down version information and trigger a separate Ajax call on
the client.
Test Plan:
Created a client pool and connected it to the server, with election and failover
apparently working correctly.
Reviewed By: aran
Reviewers: Girish, aran, jungejason, tuomaspelkonen, davidrecordon
Commenters: Girish, davidrecordon
CC: aran, epriestley, Girish, davidrecordon
Differential Revision: 284
2011-05-15 23:05:02 +02:00
|
|
|
}
|
|
|
|
|
2012-03-07 05:14:03 +01:00
|
|
|
public function log(message:String):void {
|
|
|
|
ExternalInterface.call('console.log', message);
|
Aphlict, simple notification server
Summary:
This is purely a prototype at the moment, but the basic functionality sort of
works.
I'm not sure how far I want to go with this but I think we might be able to get
somewhere without it being gross.
The idea here is to build a notification server WITHOUT using Comet, since Comet
is extremely difficult and complicated.
Instead, I use Flash on the client. LocalConnection allows flash instances to
talk to each other and connect() can be used as a locking primitive. This allows
all the instances to elect a master instance in a race-safe way. The master is
responsible for opening a single connnection to the server.
On the server, I use Node.js since PHP is pretty unsuitable for this task.
See Github Issue #3: https://github.com/facebook/phabricator/issues/3
One thing I need to figure out next is if I can reasonably do SSL/TSL over Flash
(it looks like I can, in theory, with the as3crypto library) or if the server
needs to just send down version information and trigger a separate Ajax call on
the client.
Test Plan:
Created a client pool and connected it to the server, with election and failover
apparently working correctly.
Reviewed By: aran
Reviewers: Girish, aran, jungejason, tuomaspelkonen, davidrecordon
Commenters: Girish, davidrecordon
CC: aran, epriestley, Girish, davidrecordon
Differential Revision: 284
2011-05-15 23:05:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|