- Go 1
- MySQL
- Mercurial (to download dependencies)
bin/gopush_server
$ cp config.json.sample config.json # first time only
$ vim config.json
Use make
and make install
as usual. On the developer machines, make
is enough. The server executable will be under bin
. Automatic tests will run on build.
The service will create a table called APIToken
on its first launch.
By default, testing skips the MySQL tests. If you want to test with MySQL, use the following command line switches:
- mysqluser: MySQL user name
- mysqlpass: MySQL password
- mysqldbname: Name of the MySQL database to use.
Alternatively, you can create a file called mysql.json
in the test directory, which contains the MySQL credentials (see below the configuration format).
-
Start the main server
-
Add a user through http://localhost:8080/admin
-
Copy its private key to a file, say
privkey.pem
-
Run the test client and add a notification center
$ GOPATH="`pwd`" go run src/testclient/testclient.go --privkey=privkey.pem --mail="[email protected]" --centername="test" --action=new
This will return the name of the newly created notification center. It is now${mail}____${centername}
, but it can be changed any time in the future! -
Start the test server to load the test console
$ GOPATH="`pwd`" go run src/testserver/testserver.go --address=localhost:8080/listen
-
Open http://localhost:8081/?center=$NAME_OF_THE_CENTER_YOU_CREATED
-
You can now send notifications
$ GOPATH="`pwd`" go run src/testclient/testclient.go --privkey=privkey.pem --mail="[email protected]" --centername="test" --action=notify --message=test
Go uses very lightweight thread-like channels, called goroutines. These are multiplexed between real threads. The way now the scheduler works, you have to manually specify how many threads you want to use. The service currently defaults to the number of logical CPUs available. If you want to override this, use the --procs=N
command line flag to manually specify how many CPUs the service can use.
The reason why it's important to run it as much logical CPUs as possible is the way how the goroutines gets scheduled. When a goroutine runs, the thread works on that only one goroutine. When the goroutine gets blocked, the scheduler switches to the next goroutine. When a goroutine is running, it won't be sent to the background (unlike processes or threads). It's important to realize that if only one CPU gets utilized, and one goroutine is running for a longer time (generating a key for example), all other goroutines are blocked. To achieve really good and reliable response times, it's important to use as much CPUs as possible.
There are two ways for a client to get the notifications.
The URL is /listen?center=$CENTERNAME
. The $CENTERNAME
is returned from the service when creating a new notification center (see below).
Javascript example:
if (window["WebSocket"]) {
conn = new WebSocket("ws://localhost:8080/listen?center=$CENTERNAME");
conn.onclose = function(evt) {
appendLog($("<div><b>Connection closed.</b></div>"))
}
conn.onmessage = function(evt) {
appendLog($("<div/>").text(evt.data))
}
} else {
appendLog($("<div><b>Your browser does not support WebSockets.</b></div>"))
}
For older browsers or clients, it might be a good idea to create a loop in JavaScript which checks a given URL for changes.
This method has a serious limitation: only the latest state is available. If the notifications gets produced really fast, this way the client might miss some.
The URL is /ping?center=$CENTERNAME
To circumvent the cross domain policy, this method also supports JSONP. To use it, add the callback=
parameter to the request.
To create, delete notification centers and send messages through them, you have to send POST requests to the service. All POST requests has to be signed. The signing header is:
Authorization: GoPush $RSA_SIGNATURE_HEX_ENCODED
All requests must have a GET parameter called mail
which identifies the user.
The signature is from an RSA key, generated by the service (on the /admin
page).
The signature format is RSA PKCS#1 v1.5. The hash method is SHA1.
POST /newcenter?mail=$MAIL
The body is the identifier of the new notification center.
Response: the name of the service. This name will be used with the clients to get updates from this notification center.
POST /removecenter?mail=$MAIL
The body is the identifier of the notification center.
Response: nothing, just 200 on success.
POST /notify?mail=$MAIL¢er=$CENTER_ID
The body is the notification message.
Response: nothing just 200 on success.
- address (string) The address where the server will listen to. Use hostname:port or :port to listen on all hostnames.
- dbname (string) Name of the MySQL database.
- dbuser (string) Username for the MySQL database.
- dbpass (string) Password of the MySQL user.
- certfile (string) Absolute path to the SSL certificate file. Leave empty if you don't want to use SSL.
- keyfile (string) Absolute path to the SSL private key. Leave empty if you don't want to use SSL.
- adminuser (string) Administrator username for the admin page.
- adminpass (string) Administrator password for the admin page.
- timeout (integer) After a given timeout (in seconds), notification centers will be killed. Set it to 0 to disable this behavior.
- usercache (boolean) Use an internal cache for the mail address => public key mappings. Recommended to be turned on, especially in production.
- broadcastbuffer (integer) Buffer size for the broadcasting channel. Set it to a lower value if you have a lot of notification centers with very few messages. Set it to a higher value if you have a lot of clients and a lot of messages.
- extralogging (boolean) Turns on very verbose logging. It can be really helpful for development, but turn it off in production.
- redirectmainpage (string) The main page of the service is a 404 page, which is not a really nice thing. By setting this variable, the service will redirect its main page.