httpwsd.js 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814
  1. /*******************************************************************************
  2. AToMPM - A Tool for Multi-Paradigm Modelling
  3. Copyright (c) 2011 Raphael Mannadiar (raphael.mannadiar@mail.mcgill.ca)
  4. This file is part of AToMPM.
  5. AToMPM is free software: you can redistribute it and/or modify it under the
  6. terms of the GNU Lesser General Public License as published by the Free Software
  7. Foundation, either version 3 of the License, or (at your option) any later
  8. version.
  9. AToMPM is distributed in the hope that it will be useful, but WITHOUT ANY
  10. WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
  11. PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.
  12. You should have received a copy of the GNU Lesser General Public License along
  13. with AToMPM. If not, see <http://www.gnu.org/licenses/>.
  14. *******************************************************************************/
  15. /*********************************** IMPORTS **********************************/
  16. var _util = require("util"),
  17. _fs = require('fs'),
  18. _http = require('http'),
  19. _path = require('path'),
  20. _url = require('url'),
  21. _utils = require('./utils'),
  22. _sio = require('socket.io'),
  23. _cp = require('child_process'),
  24. _fspp = require('./___fs++'),
  25. _duri = require('./___dataurize');
  26. /*********************************** GLOBALS **********************************/
  27. /* an array of WebWorkers
  28. ... each has its own mmmk instance */
  29. var workers = new Array();
  30. /* an array of response objects
  31. ... for workers to write on when they complete requests */
  32. var responses = new Array();
  33. /* a map of worker ids to socket.io socket session ids
  34. ... each socket is registered to exactly one worker
  35. ... several sockets may be registered to the same worker */
  36. var workerIds2socketIds = {};
  37. /************************************ UTILS ***********************************/
  38. /** Syntactic sugar to build and send HTTP responses **/
  39. function __respond(response, statusCode, reason, data, headers)
  40. {
  41. response.writeHead(
  42. statusCode,
  43. reason,
  44. (headers || {'Content-Type': 'text/plain'}));
  45. var encoding =
  46. (headers &&
  47. (headers['Content-Type'].match(/image/) ||
  48. headers['Content-Type'].match(/pdf/) ||
  49. headers['Content-Type'].match(/zip/) ) ?
  50. 'binary' :
  51. 'utf8'),
  52. content = reason || data;
  53. if( _utils.isObject(content) )
  54. response.end(_utils.jsons(content,null,'\t'), encoding);
  55. else
  56. response.end(content, encoding);
  57. }
  58. /** Syntactic sugar to build and send a socket.io message **/
  59. function __send(socket, statusCode, reason, data, headers)
  60. {
  61. //headers['Access-Control-Allow-Origin'] = 'http://raven10.kicks-ass.net:8080';
  62. socket.json.emit('message',
  63. {'statusCode':statusCode,
  64. 'reason':reason,
  65. 'headers':(headers || {'Content-Type': 'text/plain'}),
  66. 'data':data});
  67. }
  68. /************************************ LOGIC ***********************************/
  69. var httpserver = _http.createServer(
  70. function(req, resp)
  71. {
  72. var url = _url.parse(req.url,true);
  73. url.pathname = decodeURI(url.pathname);
  74. /* serve client */
  75. if( req.method == 'GET' && url.pathname == '/atompm' )
  76. _fs.readFile('./client/atompm.html', 'utf8',
  77. function(err, data)
  78. {
  79. if(err)
  80. __respond(resp,500,String(err));
  81. else
  82. __respond(resp,200,'',data,{'Content-Type': 'text/html'});
  83. });
  84. else if( req.method == 'GET' && url.pathname == '/favicon.png' )
  85. _fs.readFile('./favicon.png', 'binary',
  86. function(err, data)
  87. {
  88. if(err)
  89. __respond(resp,500,String(err));
  90. else
  91. __respond(resp,200,'',data,{'Content-Type': 'image/png'});
  92. });
  93. /* provide an interface to the unfortunately unavoidable dataurize
  94. module which returns data URIs for resources at arbitrary URLs */
  95. else if( req.method == 'GET' && url.pathname == '/datauri' )
  96. {
  97. var target = _url.parse(decodeURI(url['query']['target']));
  98. _duri.dataurize(
  99. target,
  100. function(err,datauri)
  101. {
  102. if(err)
  103. __respond(resp,500,_utils.jsons(err));
  104. else
  105. __respond(resp,200,'',datauri);
  106. });
  107. }
  108. /* serve metamodels, buttons models and their icons */
  109. else if( req.method == 'GET' &&
  110. (url.pathname.match(/\.metamodel$/) ||
  111. url.pathname.match(/\.buttons.model$/) ||
  112. url.pathname.match(/\.icon\.png$/i)) )
  113. {
  114. var isIcon = url.pathname.match(/\.icon\.png$/i);
  115. _fs.readFile('./users/'+url.pathname, (isIcon ? 'binary' : 'utf8'),
  116. function(err, data)
  117. {
  118. if(err)
  119. __respond(resp,500,String(err));
  120. else
  121. {
  122. var contentType =
  123. (isIcon ?
  124. {'Content-Type': 'image/png'} :
  125. {'Content-Type': 'application/json'});
  126. __respond(resp,200,'',data,contentType);
  127. }
  128. });
  129. }
  130. /* serve ordinary files (e.g., js includes, images, css)
  131. NOTE:: distinguish between atompm images (e.g., grid background,
  132. filebrowser icons) and CS/Images */
  133. else if( req.method == 'GET' &&
  134. (url.pathname.match(/\.html$/) ||
  135. url.pathname.match(/\.css$/) ||
  136. url.pathname.match(/\.js$/) ||
  137. url.pathname.match(/\.pdf$/) ||
  138. url.pathname.match(/\.png$/i) ||
  139. url.pathname.match(/\.jpg$/i) ||
  140. url.pathname.match(/\.jpeg$/i) ||
  141. url.pathname.match(/\.gif$/i) ||
  142. url.pathname.match(/\.svg$/i)) )
  143. {
  144. var isImage = url.pathname.match(/\.png$/i) ||
  145. url.pathname.match(/\.jpg$/i) ||
  146. url.pathname.match(/\.jpeg$/i) ||
  147. url.pathname.match(/\.gif$/i) ||
  148. url.pathname.match(/\.svg$/i),
  149. isText = ! isImage && ! url.pathname.match(/\.pdf$/);
  150. if( isImage && ! url.pathname.match(/^\/client\/media\//) )
  151. url.pathname = '/users/'+url.pathname;
  152. _fs.readFile('.'+url.pathname, (isText ? 'utf8' : 'binary'),
  153. function(err, data)
  154. {
  155. if(err)
  156. __respond(resp,500,String(err));
  157. else
  158. {
  159. var contentType =
  160. (url.pathname.match(/\.html$/) ?
  161. {'Content-Type': 'text/html'} :
  162. url.pathname.match(/\.css$/) ?
  163. {'Content-Type': 'text/css'} :
  164. url.pathname.match(/\.js$/) ?
  165. {'Content-Type': 'application/javascript'} :
  166. url.pathname.match(/\.pdf$/) ?
  167. {'Content-Type': 'application/pdf'} :
  168. url.pathname.match(/\.png$/i) ?
  169. {'Content-Type': 'image/png'} :
  170. url.pathname.match(/\.jpg$/i) ||
  171. url.pathname.match(/\.jpeg$/i) ?
  172. {'Content-Type': 'image/jpeg'} :
  173. url.pathname.match(/\.gif$/i) ?
  174. {'Content-Type': 'image/gif'} :
  175. url.pathname.match(/\.svg$/i) ?
  176. {'Content-Type': 'image/svg+xml'} :
  177. undefined);
  178. __respond(resp,200,'',data,contentType);
  179. }
  180. });
  181. }
  182. /* serve encrypted user password */
  183. else if( req.method == 'GET' && url.pathname == '/passwd' )
  184. _fs.readFile('./users/'+url['query']['username']+'/passwd', 'utf8',
  185. function(err, data)
  186. {
  187. if(err)
  188. __respond(resp,500,String(err));
  189. else
  190. __respond(resp,200,'',data,{'Content-Type': 'text/html'});
  191. });
  192. /* create new user
  193. 1. make sure user doesn't already exist
  194. 2. make a new copy of ./users/(default)
  195. 3. create password file */
  196. else if( req.method == 'POST' && url.pathname == '/user' )
  197. {
  198. var userdir = './users/'+url['query']['username'];
  199. _path.exists(userdir,
  200. function(exists)
  201. {
  202. if( exists )
  203. {
  204. __respond(resp,500,'username already exists');
  205. return;
  206. }
  207. _fspp.cp('./users/(default)/',userdir,
  208. function(err, stdout, stderr)
  209. {
  210. if( err )
  211. {
  212. __respond(resp,500,String(err));
  213. return;
  214. }
  215. _fs.writeFile(
  216. userdir+'/passwd',
  217. url['query']['password'],
  218. function(err)
  219. {
  220. if( err )
  221. __respond(resp,500,String(err));
  222. else
  223. __respond(resp,200);
  224. });
  225. });
  226. });
  227. }
  228. /* serve [a subset of] user preferences */
  229. else if( req.method == 'GET' && url.pathname.match(/prefs$/) )
  230. _fs.readFile('./users/'+url.pathname, 'utf8',
  231. function(err, data)
  232. {
  233. if(err)
  234. __respond(resp,500,String(err));
  235. else if( url['query']['subset'] == undefined )
  236. __respond(resp,200,'',data);
  237. else
  238. try
  239. {
  240. __respond(
  241. resp,
  242. 200,
  243. '',
  244. _utils.splitDict(
  245. _utils.jsonp(data),
  246. _utils.jsonp(url['query']['subset'])));
  247. }
  248. catch(err) {__respond(resp,500,String(err));}
  249. });
  250. /* update user preferences
  251. 1 retrieve all post data
  252. 2 read prefs file from disk
  253. 3 apply changes
  254. 4 write updated prefs to disk */
  255. else if( req.method == 'PUT' && url.pathname.match(/prefs$/) )
  256. {
  257. var reqData = '';
  258. req.addListener("data", function(chunk) {reqData += chunk;});
  259. req.addListener("end",
  260. function()
  261. {
  262. _fs.readFile('./users/'+url.pathname, 'utf8',
  263. function(err, prefs)
  264. {
  265. if(err)
  266. __respond(resp,500,String(err));
  267. else
  268. {
  269. try
  270. {
  271. prefs = _utils.jsonp(prefs);
  272. reqData = _utils.jsonp(reqData);
  273. }
  274. catch(err)
  275. {
  276. __respond(resp,500,String(err));
  277. return;
  278. }
  279. for( var pref in reqData )
  280. prefs[pref]['value'] = reqData[pref];
  281. _fs.writeFile(
  282. './users/'+url.pathname,
  283. _utils.jsons(prefs,null,'\t'),
  284. function(err, data)
  285. {
  286. if(err)
  287. __respond(resp,500,String(err));
  288. else
  289. __respond(resp,200);
  290. });
  291. }
  292. });
  293. });
  294. }
  295. /* delete specified file/folder */
  296. else if( req.method == 'DELETE' && url.pathname.match(/\.(file|folder)$/) )
  297. {
  298. if (url.pathname.match('_Trash_')) {
  299. __respond(resp,500,"cannot remove trash!");
  300. } else {
  301. var matches = url.pathname.match(/^\/(.*?)\/(.*\/)?(.*)\.(file|folder)$/),
  302. username = matches[1],
  303. folder = matches[2] || '',
  304. fname = matches[3],
  305. userdir = './users/'+username+'/',
  306. ondelete =
  307. function(err, stdout, stderr)
  308. {
  309. if( err )
  310. __respond(resp,500,String(err));
  311. else
  312. __respond(resp,200);
  313. },
  314. deletef =
  315. function(response)
  316. {
  317. var newname = userdir+'_Trash_/'+folder+fname;
  318. if (_fs.existsSync(newname)) {
  319. if (url.pathname.match(/\.folder$/)) {
  320. // http://stackoverflow.com/questions/18052762/remove-directory-which-is-not-empty
  321. var deleteFolderRecursive = function(path) {
  322. if( _fs.existsSync(path) ) {
  323. _fs.readdirSync(path).forEach(function(file,index){
  324. var curPath = path + "/" + file;
  325. if(_fs.lstatSync(curPath).isDirectory()) { // recurse
  326. deleteFolderRecursive(curPath);
  327. } else { // delete file
  328. _fs.unlinkSync(curPath);
  329. }
  330. });
  331. _fs.rmdirSync(path);
  332. }
  333. };
  334. deleteFolderRecursive(newname);
  335. } else {
  336. _fs.unlink(newname);
  337. }
  338. }
  339. _fspp.mv(userdir+folder+fname,userdir+'_Trash_/'+folder,ondelete);
  340. };
  341. _fs.exists(userdir+'_Trash_/'+folder,
  342. function(exists)
  343. {
  344. if( ! exists )
  345. _fspp.mkdirs(userdir+'_Trash_/'+folder,deletef);
  346. else {
  347. deletef();
  348. }
  349. });
  350. }
  351. }
  352. /* create folder */
  353. else if( req.method == 'POST' && url.pathname.match(/\.folder$/) )
  354. {
  355. var matches = url.pathname.match(/^\/(.*?)\/(.*)\.folder$/),
  356. username = matches[1],
  357. folder = matches[2],
  358. userdir = './users/'+username+'/',
  359. oncreate =
  360. function(err, stdout, stderr)
  361. {
  362. if( err )
  363. __respond(resp,500,String(err));
  364. else
  365. __respond(resp,200);
  366. };
  367. _fs.exists(userdir+folder,
  368. function(exists)
  369. {
  370. if( ! exists )
  371. _fspp.mkdirs(userdir+folder,oncreate);
  372. else {
  373. oncreate("folder " + folder + " already exists");
  374. }
  375. });
  376. }
  377. /* rename file/folder (or move) */
  378. else if( req.method == 'PUT' && url.pathname.match(/\.(folder|file)$/) )
  379. {
  380. req.setEncoding('utf8');
  381. var data = '';
  382. req.addListener("data", function(chunk) {data += chunk;});
  383. req.addListener("end",
  384. function() {
  385. data = _utils.jsonp(data);
  386. if (data.match(/^\//)) {
  387. // move
  388. var matches = url.pathname.match(/^\/(.*?)\/(.*\/)?(.*)\.(file|folder)$/),
  389. username = matches[1],
  390. folder = matches[2] || '',
  391. fname = matches[3],
  392. userdir = './users/'+username,
  393. onmove =
  394. function(err, stdout, stderr)
  395. {
  396. if( err )
  397. __respond(resp,500,String(err));
  398. else
  399. __respond(resp,200);
  400. };
  401. _fspp.mv(userdir+"/"+folder+fname,userdir+data,onmove)
  402. } else {
  403. // rename
  404. var matches = url.pathname.match(/^\/(.*?)\/(.*\/)?(.*)\.(file|folder)$/),
  405. username = matches[1],
  406. folder = matches[2] || '',
  407. fname = matches[3],
  408. userdir = './users/'+username+'/',
  409. onrename =
  410. function(err, stdout, stderr)
  411. {
  412. if( err )
  413. __respond(resp,500,String(err));
  414. else
  415. __respond(resp,200);
  416. };
  417. _fs.rename(userdir+folder+fname,userdir+folder+data,onrename)
  418. }
  419. }
  420. )
  421. }
  422. else if (req.method == 'POST' && url.pathname.match(/\.formalism$/)) {
  423. // create new formalism
  424. var matches = url.pathname.match(/^(.*)\/(.*)\.formalism$/),
  425. username = matches[1],
  426. formalism = matches[2],
  427. userdir = './users/'+username+"/",
  428. oncreatefolder =
  429. function(err, stdout, stderr)
  430. {
  431. if( err )
  432. __respond(resp,500,String(err));
  433. else {
  434. _fs.createReadStream(userdir+"Formalisms/__Templates__/MetamodelTemplate.model").pipe(_fs.createWriteStream(userdir+"Formalisms/"+formalism+"/"+formalism+".model"));
  435. _fs.createReadStream(userdir+"Formalisms/__Templates__/ConcreteSyntaxTemplate.model").pipe(_fs.createWriteStream(userdir+"Formalisms/"+formalism+"/"+formalism+".defaultIcons.model"));
  436. _fs.createReadStream(userdir+"Formalisms/__Templates__/T_TransformationTemplate.model").pipe(_fs.createWriteStream(userdir+"Formalisms/"+formalism+"/OperationalSemantics/T_OperationalSemantics.model"));
  437. _fs.createReadStream(userdir+"Formalisms/__Templates__/T_TransformationTemplate.model").pipe(_fs.createWriteStream(userdir+"Formalisms/"+formalism+"/TranslationalSemantics/T_TranslationalSemantics.model"));
  438. __respond(resp,200);
  439. }
  440. };
  441. _fs.mkdir(userdir+"Formalisms/"+formalism,function(err, stdout, stderr) {
  442. if( err )
  443. __respond(resp,500,String(err));
  444. else {
  445. _fs.mkdirSync(userdir+"Formalisms/"+formalism+"/OperationalSemantics");
  446. _fs.mkdir(userdir+"Formalisms/"+formalism+"/TranslationalSemantics", oncreatefolder);
  447. }
  448. });
  449. }
  450. else if (req.method == 'POST' && url.pathname.match(/\.transformation$/)) {
  451. // create new transformation
  452. var matches = url.pathname.match(/^\/(.*?)\/(.*)\.transformation$/),
  453. username = matches[1],
  454. userdir = './users/'+username+"/";
  455. _fs.createReadStream(userdir+"Formalisms/__Templates__/T_TransformationTemplate.model").pipe(_fs.createWriteStream('./users/'+url.pathname.slice(0, -(".transformation".length))));
  456. __respond(resp,200);
  457. }
  458. else if (req.method == 'POST' && url.pathname.match(/\.rule$/)) {
  459. // create new rule
  460. var matches = url.pathname.match(/^\/(.*?)\/(.*)\.rule$/),
  461. username = matches[1],
  462. userdir = './users/'+username+"/";
  463. _fs.createReadStream(userdir+"Formalisms/__Templates__/R_RuleTemplate.model").pipe(_fs.createWriteStream('./users/'+url.pathname.slice(0, -(".rule".length))));
  464. __respond(resp,200);
  465. }
  466. /* extract user-uploaded archive to specified folder
  467. 1. read in all data
  468. 2. make sure destination exists and is a directory
  469. 3. write data to temp file (upload###.zip)
  470. 4. extract temp file and remove it
  471. NOTE:: it's not clear why (despite hours of googling) but the
  472. "req.setEncoding('utf8')" statement makes the difference
  473. between retrieving correct and corrupted (when non-text
  474. files in zip) data */
  475. else if( req.method == 'PUT' && url.pathname.match(/\.file$/) )
  476. {
  477. req.setEncoding('utf8');
  478. var reqData = '',
  479. tmpzip = 'upload'+Date.now()+'.zip',
  480. destdir = './users/'+url.pathname.match(/(.*)\.file$/)[1]+'/';
  481. req.addListener("data", function(chunk) {reqData += chunk;});
  482. req.addListener("end",
  483. function()
  484. {
  485. _fs.stat(destdir,
  486. function(err,stats)
  487. {
  488. if( err )
  489. __respond(resp,404,String(err));
  490. else if( ! stats.isDirectory() )
  491. __respond(resp,404,
  492. 'destination is not a directory :: '+destdir);
  493. else
  494. _fs.writeFile(
  495. destdir+tmpzip,eval('('+reqData+')'),
  496. 'binary',
  497. function(err)
  498. {
  499. _cp.exec('cd '+destdir+'; unzip -o '+tmpzip,
  500. function(err, stdout, stderr)
  501. {
  502. if( err )
  503. __respond(resp,500,String(err));
  504. else
  505. __respond(resp,200);
  506. _fs.unlink(destdir+tmpzip);
  507. });
  508. });
  509. });
  510. });
  511. }
  512. /* serve specified file/folder within a zip file */
  513. else if( req.method == 'GET' && url.pathname.match(/\.file$/) )
  514. {
  515. var matches = url.pathname.match(/^\/(.*?)\/(.*)\.file$/),
  516. username = matches[1],
  517. fname = './'+matches[2],
  518. userdir = './users/'+username+'/',
  519. tmpzip = 'download'+Date.now()+'.zip';
  520. _fs.exists(userdir+fname,
  521. function(exists)
  522. {
  523. if( ! exists )
  524. __respond(resp,404,
  525. 'requested file/folder does not exist :: '+fname);
  526. else
  527. _cp.exec('cd '+userdir+'; zip -r '+tmpzip+' "'+fname+'"',
  528. function(err, stdout, stderr)
  529. {
  530. if( err )
  531. __respond(resp,500,String(err));
  532. else
  533. _fs.readFile(userdir+tmpzip,'binary',
  534. function(err, data)
  535. {
  536. __respond(resp,200,'',data,
  537. {'Content-Type':'application/zip',
  538. 'Content-Disposition':
  539. 'attachment; filename="'+tmpzip+'"'});
  540. _fs.unlink(userdir+tmpzip);
  541. });
  542. });
  543. });
  544. }
  545. /* serve list of all files */
  546. else if( req.method == 'GET' &&
  547. url.pathname.match(/^\/.+\/filelist$/) )
  548. {
  549. var matches = url.pathname.match(/^\/(.+)\/filelist$/);
  550. _fspp.findfiles('./users/'+matches[1],
  551. function(err, stdout, stderr)
  552. {
  553. if( err )
  554. __respond(resp,404,String(err));
  555. else
  556. __respond(resp,200,'',stdout);
  557. });
  558. }
  559. /* spawn new worker */
  560. else if( (url.pathname == '/csworker' || url.pathname == '/asworker')
  561. && req.method == 'POST' )
  562. {
  563. /* setup and store new worker */
  564. var worker = _cp.fork(_path.join(__dirname, '__worker.js')),
  565. wid = workers.push(worker)-1;
  566. workerIds2socketIds[wid] = [];
  567. worker.on('message',
  568. function(msg)
  569. {
  570. /* push changes (if any) to registered sockets... even empty
  571. changelogs are pushed to facilitate sequence number-based
  572. ordering */
  573. if( msg['changelog'] != undefined )
  574. {
  575. var _msg = {'changelog':msg['changelog'],
  576. 'sequence#':msg['sequence#'],
  577. 'hitchhiker':msg['hitchhiker']};
  578. workerIds2socketIds[wid].forEach(
  579. function(sid)
  580. {
  581. __send(
  582. wsserver.sockets.sockets[sid],
  583. undefined,
  584. undefined,
  585. _msg);
  586. });
  587. }
  588. /* respond to a request */
  589. if( msg['respIndex'] != undefined )
  590. __respond(
  591. responses[msg['respIndex']],
  592. msg['statusCode'],
  593. msg['reason'],
  594. JSON.stringify(
  595. {'headers':
  596. (msg['headers'] ||
  597. {'Content-Type': 'text/plain'}),
  598. 'data':msg['data'],
  599. 'sequence#':msg['sequence#']}),
  600. {'Content-Type': 'application/json'});
  601. });
  602. worker.send(
  603. {'workerType':url.pathname,
  604. 'workerId':wid});
  605. /* respond worker id (used to identify associated worker) */
  606. __respond(
  607. resp,
  608. 201,
  609. '',
  610. ''+wid);
  611. return;
  612. }
  613. /* check for worker id and it's validity */
  614. else if( url['query'] == undefined ||
  615. url['query']['wid'] == undefined )
  616. __respond(resp, 400, 'missing worker id');
  617. else if( workers[url['query']['wid']] == undefined )
  618. __respond(resp, 400, 'invalid worker id :: '+url['query']['wid']);
  619. /* save resp object and forward request to worker (if request is PUT or
  620. POST, recover request data first)
  621. TBI:: only registered sockets should be allowed to speak to worker
  622. ... one way of doing this is forcing request urls to contain
  623. cid=socket.id## */
  624. else if( req.method == 'PUT' || req.method == 'POST' )
  625. {
  626. var reqData = '';
  627. req.addListener("data", function(chunk) {reqData += chunk;});
  628. req.addListener("end",
  629. function()
  630. {
  631. workers[url['query']['wid']].send(
  632. {'method':req.method,
  633. 'uri':url.pathname,
  634. 'reqData':(reqData == '' ?
  635. undefined :
  636. eval('('+reqData+')')),
  637. 'uriData':url['query'],
  638. 'respIndex':responses.push(resp)-1});
  639. });
  640. }
  641. else
  642. workers[url['query']['wid']].send(
  643. {'method':req.method,
  644. 'uri':url.pathname,
  645. 'uriData':url['query'],
  646. 'respIndex':responses.push(resp)-1});
  647. });
  648. httpserver.listen(8124);
  649. var wsserver = _sio.listen(httpserver);
  650. wsserver.configure(
  651. function()
  652. {
  653. wsserver.set('log level',2);
  654. });
  655. wsserver.sockets.on('connection',
  656. function(socket)
  657. {
  658. /* unregister this socket from the specified worker ... when a worker
  659. has no more registered sockets, terminate it */
  660. function unregister(wid)
  661. {
  662. var i = workerIds2socketIds[wid].indexOf(socket.id)
  663. if( i == -1 )
  664. __send(socket,403,'already unregistered from worker');
  665. else
  666. {
  667. workerIds2socketIds[wid].splice(i,1);
  668. if( workerIds2socketIds[wid].length == 0 )
  669. {
  670. workers[wid].kill();
  671. workers[wid] = undefined;
  672. delete workerIds2socketIds[wid];
  673. }
  674. __send(socket,200);
  675. }
  676. }
  677. /* onmessage : on reception of data from client */
  678. socket.on('message',
  679. function(msg/*{method:_,url:_}*/)
  680. {
  681. var url = _url.parse(msg.url,true);
  682. /* check for worker id and it's validity */
  683. if( url['query'] == undefined ||
  684. url['query']['wid'] == undefined )
  685. return __send(socket,400,'missing worker id');
  686. var wid = url['query']['wid'];
  687. if( workers[wid] == undefined )
  688. __send(socket,400,'unknown worker id :: '+wid);
  689. /* register socket for requested worker */
  690. else if( msg.method == 'POST' &&
  691. url.pathname.match(/changeListener$/) )
  692. {
  693. if( workerIds2socketIds[wid].indexOf(socket.id) > -1 )
  694. __send(socket,403,'already registered to worker');
  695. else
  696. {
  697. workerIds2socketIds[wid].push(socket.id);
  698. __send(socket,201);
  699. }
  700. }
  701. /* unregister socket for requested worker */
  702. else if( msg.method == 'DELETE' &&
  703. url.pathname.match(/changeListener$/) )
  704. unregister(wid);
  705. /* unsupported request */
  706. else
  707. __send(socket,501);
  708. });
  709. /* ondisconnect : on disconnection of socket */
  710. socket.on('disconnect',
  711. function()
  712. {
  713. for( var wid in workerIds2socketIds )
  714. for( var i in workerIds2socketIds[wid] )
  715. if( workerIds2socketIds[wid][i] == socket.id )
  716. {
  717. unregister(wid);
  718. return;
  719. }
  720. });
  721. });