// A bot which responds to `jarvis` with a box const Space = require('../space'); const Socket = require('../socket'); const ms = require('../utils/measurespace'); const vec = require('../utils/vec'); const comb = require('../utils/comb'); const Search = require('../tools/search'); const sched = require('../tools/schedule'); const ri = require('../utils/rectintersect'); const id = require('../utils/ident'); const maketiles = require('../utils/maketiles'); //// The queue of all writes to be sent to the server @ 200chars/1000ms var writes = new sched.Queue(1000, 200, (elems) => main.write(elems)); var main = new Socket(); // See ident.js for further documentation, but this basically sets up init functions id(main, initOnce, init, deinit); var sender; function init(send){ sender = send; // tileUpdates require knowledge of the sender setTimeout(()=>{writes.enable()},1000); // Would fail if a write were added because it would be within a second of cursor send } function deinit(){ writes.disable(); } function initOnce(){ timect(); main.on('tileUpdate', tileHandler); // Should only be active after the "control space" of the notification has been established } //// Management utilities // When a tile changes, processes data into usable form and runs every function in `funcs` w/ that data. // Note that functions can edit inputs function tileHandler(send, source, tiles){ if (send == sender) return; let tiledata = maketiles(tiles); funcs.forEach(func => { func(send, tiledata.tilespaces, tiledata.locs); }); } //// "userspace" functions for responses, like to tileUpdates var funcs = [protectArea, detectPrompt]; var command = 'jarvis' var sig = 'feynmansfedora' var notifsrc = '\ \n\ For a node.js YWOT api: \n\ & git.hrhr.dev/ywot-clean \n\ & Try `COMMAND` today. Uptime: UPTIME. \n\ & Called: JARVISx ~SIGNATURE \n\ ' notifsrc = notifsrc.replace('COMMAND', command).replace('SIGNATURE', sig); var minsUp = 0; var callct = 0; var ctrl; function notifRefresh(){ let newnotif = new Space(); newnotif.loc = [-6, 20]; text = notifsrc; text = text.replace('UPTIME', Math.floor(minsUp/60) + 'h' + minsUp%60 + 'm'); text = text.replace('JARVIS', callct); newnotif.adhoc(text); let diff = newnotif.copy(); if (typeof notif !== 'undefined') diff.comb(notif, comb.sub); notif = newnotif; ctrl = ms(notif); return (diff.towrite()); } function timect(){ writes.enqueue(notifRefresh()); minsUp++; setTimeout(timect, 60*1000); } function protectArea(send, tiles, locs){ let tilesize = [8, 16]; // This may be exportable to an external util/tool for (loc of locs){ let tile = tiles[loc]; let origloc = loc; loc = vec.tileToChar(loc); if (ri(ctrl, [loc, vec.add(loc, tilesize)])){ // Write diffs from the tile let diff = notif.copy(); // Acts as a window for tile (only see relevant data) diff.comb(tile, comb.flip(comb.unmask)); diff.comb(tile, comb.sub); writes.enqueue(diff.towrite()) // Mask the tile let newtile = notif.copy(); newtile.comb(tile, comb.mask); tiles[origloc] = newtile; } } } var search = new Space(); search.adhoc('jarvis'); // The search space is the word jarvis, so whenever that's caught, a relevant function can be called. var read = new Search(search); var expire = {}; function detectPrompt(send, tiles, locs){ // tries to detect the prompt ('jarvis') and calls respond if found. for (let i=0; i 0) respond(results, send); expire[loc] = setTimeout(() => {read.del(loc); delete expire[loc];}, 30000); } } let response = new Space(); response.adhoc('yes, my liege'); var limits = []; // an array of senders within the last 5 seconds to act as a 'rate limiter' function respond(coord, send){ if (limits.indexOf(send) >= 0) return; console.log('called at', coord); callct += 1; response.loc = coord; writes.enqueue(response.towrite().concat(notifRefresh())); read.update(response); setTimeout(() => {limits.splice(limits.indexOf(send))}, 5*1000); limits.push(send); }