Skip to content

Commit

Permalink
Sockets: Go rewrite
Browse files Browse the repository at this point in the history
Work in progress. I haven't written documentation for the Go code or
confirmed whether this works as intended on Windows yet.

This turned into a pretty substantial refactor of the Node version of
sockets-related code to not only to make using Go child processes
possible, but to optimize it and make unit testing of it and any code
dependent on it possible to write entirely synchronously. sockets.js
and sockets-workers.js are now written in Typescript, though they won't
be able to be transpiled until after Config, Users, Dnsbl, and Monitor
work with it as well.

Fixes smogon#2943
  • Loading branch information
Morfent authored and Unknown committed Aug 12, 2017
1 parent 4ee1554 commit 633a112
Show file tree
Hide file tree
Showing 19 changed files with 2,330 additions and 661 deletions.
7 changes: 7 additions & 0 deletions config/config-example.js
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,13 @@ exports.proxyip = false;
// $ npm install --no-save ofe
exports.ofe = false;

// Go language - whether or not to use Go instead of Node.js to host the static
// and SockJS servers. Go is more likely to be more performant than Node.js for
// this purpose, but this should be kept set to false unless you're capable of
// debugging any issues that may arise due to the additional complexity of
// the code needed for this to run.
exports.golang = false;

// Pokemon of the Day - put a pokemon's name here to make it Pokemon of the Day
// The PotD will always be in the #2 slot (not #1 so it won't be a lead)
// in every Random Battle team.
Expand Down
35 changes: 35 additions & 0 deletions dev-tools/sockets.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
'use strict';

const {Session, SockJSConnection} = require('sockjs/lib/transport');

const chars = 'abcdefghijklmnopqrstuvwxyz1234567890-';
let sessionidCount = 0;

/**
* @return string
*/
function generateSessionid() {
let ret = '';
let idx = sessionidCount;
for (let i = 0; i < 8; i++) {
ret = chars[idx % chars.length] + ret;
idx = idx / chars.length | 0;
}
sessionidCount++;
return ret;
}

/**
* @param {string} sessionid
* @param {{options: {{}}} config
* @return SockJSConnection
*/
exports.createSocket = function (sessionid = generateSessionid(), config = {options: {}}) {
let session = new Session(sessionid, config);
let socket = new SockJSConnection(session);
socket.remoteAddress = '127.0.0.1';
socket.protocol = 'websocket';
return socket;
};

// TODO: move worker mocks here, use require('../sockets-workers').Multiplexer to stub IPC
115 changes: 97 additions & 18 deletions pokemon-showdown
Original file line number Diff line number Diff line change
Expand Up @@ -39,27 +39,106 @@ try {
);
}

if (!process.argv[2] || /^[0-9]+$/.test(process.argv[2])) {
// Start the server. We manually load app.js so it can be configured to run as
// the main module, rather than this file being considered the main module.
// This ensures any dependencies that were just installed can be found when
// running on Windows and avoids any other potential side effects of the main
// module not being app.js like it is assumed to be.
//
// The port the server should host on can be passed using the second argument
// when launching with this file the same way app.js normally allows, e.g. to
// host on port 9000:
// $ ./pokemon-showdown 9000

require('module')._load('./app', module, true);
} else switch (process.argv[2]) {
// ALlow arguments passed to the launch script to be evaluated as commands.
let [, , arg2, arg3, arg4] = process.argv;
if (arg2 && /^[0-9]$/.test(arg2)) {
switch (arg2) {
case 'generate-team':
const Dex = require('./sim/dex');
global.toId = Dex.getId;
const seed = process.argv[4] ? process.argv[4].split(',').map(Number) : undefined;
console.log(Dex.packTeam(Dex.generateTeam(process.argv[3], seed)));
break;
const seed = arg4 ? arg4.split(',').map(Number) : undefined;
console.log(Dex.packTeam(Dex.generateTeam(arg3, seed)));
process.exit(0);
default:
console.error('Unrecognized command: ' + process.argv[2]);
console.error(`Unrecognized command: ${arg2}`);
process.exit(1);
}
}

// If evaluating commands wasn't the point of running this script, let's launch
// the server.

// Check if the server is configured to use Go, and ensure the required
// environment variables and dependencies are available if that is the case

let config;
try {
config = require('./config/config');
} catch (e) {}

if (config && config.golang) {
if (!process.env.GOPATH) {
console.log('The GOPATH environment variable is not set! It is required in order to run the server using Go.');
process.exit(0);
}
if (!process.env.GOROOT) {
console.log('The GOROOT environment variable is not set! It is required in order to run the server using Go.');
process.exit(0);
}

const dependencies = ['github.com/gorilla/mux', 'github.com/igm/sockjs-go/sockjs'];
let packages = child_process.execSync('go list all', {stdio: null, encoding: 'utf8'});
for (let dep of dependencies) {
if (!packages.includes(dep)) {
console.log(`Dependency ${dep} is not installed. Fetching...`);
child_process.execSync(`go get ${dep}`, {stdio: 'inherit'});
}
}

const {GOPATH} = process.env;
let stat;
let needsSrcDir = false;
try {
stat = fs.lstatSync(path.resolve(GOPATH, 'src/github.com/Zarel'));
} catch (e) {
needsSrcDir = true;
} finally {
if (stat && !stat.isDirectory()) {
needsSrcDir = true;
}
}

if (needsSrcDir) {
try {
fs.mkdirSync(path.resolve(GOPATH, 'src/github.com/Zarel'));
} catch (e) {
console.error(`Cannot make go source directory for the sockets library files! Symlink them manually from ${__dirname} to ${path.resolve(GOPATH, 'src/github.com/Zarel/Pokemon-Showdown/')}`);
process.exit(0);
}
}

try {
stat = fs.lstatSync(path.resolve(GOPATH, 'src/github.com/Zarel/Pokemon-Showdown'));
} catch (e) {}

if (!stat || !stat.isSymbolicLink()) {
try {
// FIXME: does this even work on Windows? Check to see if `mklink /J`
// might be needed instead
fs.symlink(__dirname, path.resolve(GOPATH, 'src/github.com/Zarel/Pokemon-Showdown'));
} catch (e) {
console.error(`Cannot make go source directory for the sockets library files! Symlink them manually from ${__dirname} to ${path.resolve(GOPATH, './src/github.com/Zarel/Pokemon-Showdown/')}`);
process.exit(0);
}
}

console.log('Building Go source libs...');
try {
child_process.execSync('go install github.com/Zarel/Pokemon-Showdown/sockets', {stdio: 'inherit'});
} catch (e) {
process.exit(0);
}
}

// Start the server. We manually load app.js so it can be configured to run as
// the main module, rather than this file being considered the main module.
// This ensures any dependencies that were just installed can be found when
// running on Windows and avoids any other potential side effects of the main
// module not being app.js like it is assumed to be.
//
// The port the server should host on can be passed using the second argument
// when launching with this file the same way app.js normally allows, e.g. to
// host on port 9000:
// $ ./pokemon-showdown 9000

require('module')._load('./app', module, true);
Loading

0 comments on commit 633a112

Please sign in to comment.