webui.t
#charset "us-ascii"
/*
* Copyright (c) 2010 Michael J. Roberts. All Rights Reserved.
*
* This module defines some useful helper functions for implementing a
* TADS game with a Web-based user interface.
*/
#include <tads.h>
#include <tadsnet.h>
#include <httpsrv.h>
#include <httpreq.h>
#include <strbuf.h>
#include <file.h>
/* ------------------------------------------------------------------------ */
/*
* write a message to the system debug log
*/
#define DbgMsg(msg) t3DebugTrace(T3DebugLog, msg)
/* ------------------------------------------------------------------------ */
/*
* Session timeout settings. Times are in milliseconds.
*/
/*
* Housekeeping interval. We'll wait at least this long between
* housekeeping passes.
*/
#define HousekeepingInterval 15000
/*
* Session startup timeout. When we first start up, we'll give our first
* client this long to establish a connection. If we don't hear anything
* from a client within this interval, we'll assume that the client
* exited before it had a chance to set up its first connection, so we'll
* terminate the server.
*/
#define SessionStartupTimeout 90000
/*
* Ongoing session timeout. After we've received our first connection,
* we'll terminate the server if we go this long without any active
* connections.
*/
#define SessionTimeout 120000
/*
* Event request timeout. After an event request from a client has been
* sitting in the queue for longer than this limit, we'll send a "keep
* alive" reply to let the client know that we're still here. The client
* will just turn around and send a new request. This periodic handshake
* lets the client know we're still alive, and vice versa.
*/
#define EventRequestTimeout 90000
/*
* Client session timeout. If we haven't seen a request from a given
* client within this interval, we'll assume that the client has
* disconnected, and we'll drop the client session state on this end.
*/
#define ClientSessionTimeout 60000
/* ------------------------------------------------------------------------ */
/*
* Some handy Vector extensions
*/
modify Vector
push(ele) { append(ele); }
pop() { return getAndRemove(length()); }
shift() { return getAndRemove(1); }
getAndRemove(idx)
{
/* get the element */
local ret = self[idx];
/* remove it */
removeElementAt(idx);
/* return the popped element */
return ret;
}
clear()
{
if (length() > 0)
removeRange(1, length());
}
;
/* ------------------------------------------------------------------------ */
/*
* HTTPRequest extensions
*/
modify HTTPRequest
/*
* Send a reply, catching "socket disconnect" exceptions. In most
* cases, server objects will want to use this method rather than the
* native sendReply() so that they won't have to handle disconnect
* exceptions manually.
*
* Disconnect exceptions are common with HTTP: the protocol is
* stateless by design, so clients can close their sockets at any
* time. Most modern browsers use HTTP 1.1, which allows the client
* to maintain an open socket indefinitely and reuse it serially for
* multiple requests, but browsers tend to close these reusable
* sockets after a few minutes of inactivity.
*
* When this routine encounters a disconnected socket, it deletes the
* client session record for the request, and then otherwise ignores
* the error. This generally makes the client's socket management
* transparent to the server, since if the client is still running
* they'll just connect again with a new socket and retry any lost
* requests.
*/
sendReplyCatch(reply, contType?, stat?, headers?)
{
/* catch any errors */
try
{
/* try sending the reply */
sendReply(reply, contType, stat, headers);
}
catch (SocketDisconnectException sde)
{
/* notify the associated client session of the disconnect */
local client = ClientSession.find(self);
if (client != nil)
client.checkDisconnect();
}
}
;
/* ------------------------------------------------------------------------ */
/*
* Web UI Session object. This keeps track of miscellaneous items
* associated with the game session.
*/
transient webSession: object
/*
* Get the full URL to the given resource.
*/
getFullUrl(resname)
{
return 'http://<<server.getAddress()>>:<<
server.getPortNum()>><<resname>>';
}
/*
* Connect to the UI. By default, we ask the webMainWin object to
* establish a connection, and we save the server object internally
* for future reference.
*/
connectUI(srv)
{
/* connect the UI to the main window object */
webMainWin.connectUI(srv);
/* save a reference to the server object */
server = srv;
/*
* set the zero point for the connection time - for timeout
* purposes, the countdown starts now, since we obviously can't
* have a connection before this point
*/
lastClientTime = getTime(GetTimeTicks);
}
/*
* The session key. This identifies the server as a whole, and is
* essentially an authentication mechanism that lets clients prove
* they got our address from an authorized source (rather than just
* stumbling across it via a port scan, say). Clients must hand this
* to us on each request, either via a URL query parameter or via a
* cookie. The normal setup (via WebResourceInit) is for the client
* to send us the key as a URL parameter on the initial request, at
* which point we'll pass it back as a set-cookie, removing the need
* for the client to include the key in subsequent URLs.
*
* The key is just a random number that's long enough that an
* interloper couldn't hope to guess it. We generate this on the
* first evaluation, and it remains fixed at that point for as long
* as we're running.
*/
sessionKey = (sessionKey = generateRandomKey())
/*
* The collaborative session key. This is a secondary session key
* that allows additional users to connect to the session for
* collaborative play.
*/
collabKey = (collabKey = generateRandomKey())
/*
* Validate a session key sent from the client
*/
validateKey(req, query)
{
/* get the key, either from the query string or from a cookie */
local rkey = query['TADS_session'];
if (rkey == nil)
rkey = req.getCookie('TADS_session');
/* if there's no key, it's an error */
if (rkey == nil)
{
/* no key -> bad request */
req.sendReply(400);
return nil;
}
/* the key has to match either the main key or the collab key */
if (rkey != sessionKey && rkey != collabKey)
{
/* invalid key -> forbidden */
req.sendReply(403);
return nil;
}
/* the session key is valid */
return true;
}
/*
* The launcher's game ID. This is the ID passed from the web server
* that launched the game, to let us know how the game is identified
* in the launcher database. This is typically an IFDB TUID string.
*/
launcherGameID = nil
/*
* The launcher's user name. This is passed from the web server that
* launched the game, to let us know the host user's screen name. We
* use this as the user's default screen name in multi-user games.
*/
launcherUsername = 'Host'
/*
* The primary storage server session ID, for the user who launched
* the server. If the user who launched the game logged in to a
* cloud storage server, this is the session ID that we use to
* transact business with the server on behalf of this logged-in
* user. This token identifies and authenticates the user, but it's
* ephemeral and it's only valid for the current game server session,
* so it's not quite like a password. This is the session for the
* launch user only; if other collaborative users join, they can get
* their own session IDs that will allow them to store files under
* their own private user folders on the server.
*/
storageSID = nil
/*
* Get the collaborative player launch URL. This is a URL that the
* host can send to other players who wish to join the session as
* collaborative users.
*/
getCollabUrl()
{
/* return the main window URL */
return webSession.getFullUrl(
'<<webMainWin.vpath>>?TADS_session=<<collabKey>>');
}
/* list of active client sessions (ClientSession objects) */
clientSessions = static new transient Vector()
/* add a client session */
addClient(s)
{
clientSessions.append(s);
lastClientTime = getTime(GetTimeTicks);
everHadClient = true;
}
/* remove a client session */
removeClient(s)
{
clientSessions.removeElement(s);
lastClientTime = getTime(GetTimeTicks);
}
/* the HTTPServer object running our web session */
server = nil
/*
* Run housekeeping tasks. The network event processor calls this
* periodically to let us perform background cleanup tasks. Returns
* the system tick time of the next housekeeping run.
*/
housekeeping()
{
/*
* if it hasn't been long enough yet since the last housekeeping
* run, skip this
*/
if (getTime(GetTimeTicks) < hkTime)
return hkTime;
/* send keep-alives and check for dead client sessions */
clientSessions.forEach(function(s)
{
/* send keep-alives to aged-out event requests */
s.sendKeepAlive();
/* check to see if the client has disconnected */
s.checkDisconnect();
});
/* if there are no client sessions, check for server termination */
if (clientSessions.length() == 0)
{
/*
* There are no clients connected. If we've exceeded the
* maximum interval for running without a client, shut down
* the server. Use the initial connection time limit if
* we've never had a client, otherwise use the ongoing idle
* time limit.
*/
local limit = everHadClient
? SessionTimeout : SessionStartupTimeout;
if (getTime(GetTimeTicks) > lastClientTime + limit)
{
/* no clients - shut down */
throw new QuittingException();
}
}
else
{
/* we have a session as of right now */
lastClientTime = getTime(GetTimeTicks);
}
/* update the housekeeping timer */
return hkTime = getTime(GetTimeTicks) + HousekeepingInterval;
}
/* system time (ms ticks) of next scheduled housekeeping pass */
hkTime = 0
/* the last time we noticed that we had a client connected */
lastClientTime = 0
/* have we ever had a client connection? */
everHadClient = nil
;
/*
* Generate a random key. This returns a 128-bit random number as a hex
* string. This is designed for ephemeral identifiers, such as session
* keys.
*/
generateRandomKey()
{
/* generate a long random hex string */
return rand('xxxxxxxx%-xxxx%-xxxx%-xxxxxxxxxxxx%-xxxx');
}
/* ------------------------------------------------------------------------ */
/*
* Client session. This represents a connection to one browser (or other
* client application). Each browser client is a separate session, so we
* create one instance of this class per connected browser. Note that
* browser instances don't necessarily represent different users - a
* single user could open multiple browser windows on the same server.
*
* We identify each browser instance via a session cookie, which we
* establish when the client connects. The browser sends the cookie with
* each subsequent request, allowing us to tie the request to the browser
* session we previously set up.
*/
class ClientSession: object
construct(skey, ssid)
{
/* remember my session key and storage server SID */
storageSID = ssid;
/* note if we're a secondary (collaborative play) user */
isPrimary = (skey == webSession.sessionKey);
/* add me to the master list of sessions */
webSession.addClient(self);
/* note our last activity time */
updateEventTime();
/* create a UI preferences object */
uiPrefs = new WebUIPrefs(self);
}
/* the UI preferences object for this session */
uiPrefs = nil
/*
* The client's "screen name" - this is the user-visible name that
* we'll show other users to identify commands and chat messages
* entered by this client.
*/
screenName = ''
/* set the default screen name for a client */
setDefaultScreenName()
{
/*
* If this is the primary, the name is the launching user's
* screen name. If not, it's 'Guest N', where N is the number of
* guest connections we have.
*/
screenName = (isPrimary ? webSession.launcherUsername :
'Guest <<webSession.clientSessions.countWhich(
{x: !x.isPrimary})>>');
}
/* the client's IFDB user ID (a "TUID"), if logged in to IFDB */
ifdbTuid = nil
/* the list of pending event requests from this client */
pendingReqs = perInstance(new Vector())
/*
* The client's event queue. When a server-to-client event occurs,
* we post it to each current client's queue. When the client sends
* a get-event request, we satisfy it out of this queue.
*/
pendingEvts = perInstance(new Vector())
/*
* The client session key. This identifies the client across
* requests. We send this to the client as a cookie when they
* connect, so we get it back on each request.
*/
clientKey = perInstance(generateRandomKey())
/*
* The storage server session key for the user connected to this
* session, if any. We can have multiple users logged in to the game
* in collaborative play mode, each with their own separate storage
* server session. This allows each user to have their own private
* preference settings, saved games, etc.
*/
storageSID = nil
/*
* Am I the primary player? This is true if the player connected
* using the primary session key. Collaborative players join through
* the separate collaborative session key.
*/
isPrimary = nil
/*
* Is this session alive? When we detect that the client has
* disconnected, we'll set this to nil. When waiting for a client in
* a modal event loop, this can be used to terminate the wait if the
* client disconnects.
*/
isAlive = true
/*
* Last request time, in system ticks (ms). We use this to determine
* how long it's been since we've heard from the client, for timeout
* purposes. This is updated any time we receive a command or event
* request from the client, and each time we successfully send an
* event reply.
*/
lastEventTime = 0
/* update the last event time for this client */
updateEventTime() { lastEventTime = getTime(GetTimeTicks); }
/* class method: broadcast an event message to all connected clients */
broadcastEvent(msg)
{
/* send the event to each client in our active list */
webSession.clientSessions.forEach({ c: c.sendEvent(msg) });
}
/* send an event to this client */
sendEvent(msg)
{
/* enqueue the event, then match it to a request if possible */
pendingEvts.push(msg);
processQueues();
}
/* receive an event request from the client */
requestEvent(req)
{
/* enqueue the request, then match it to an event if possible */
pendingReqs.push(new ClientEventRequest(req));
processQueues();
}
/* flush outstanding events for this client */
flushEvents()
{
/* discard any queued events */
pendingEvts.clear();
/* send no-op replies to any pending event requests */
while (pendingReqs.length() > 0)
{
/* pull out the first request */
local req = pendingReqs.shift();
/*
* Send a reply. Ignore any errors: the client probably
* canceled their side of the socket already, so we'll
* probably get a socket error sending the reply. Since
* we're just canceling the request anyway, this is fine.
*/
try
{
/* send a no-op reply */
req.req.sendReply('<?xml version="1.0"?><noOp></noOp>',
'text/xml', 200);
/* update the successful communications time */
updateEventTime();
}
catch (Exception exc)
{
/*
* ignore errors - the client probably canceled their
* side of the socket already, so we'll probably fail
* sending the reply
*/
}
}
}
/*
* Send a keep-alive reply to each pending request from this client
* that's been waiting for longer than the timeout interval.
*
* Javascript clients in principle will wait indefinitely for an
* XmlHttpRequest to complete, but in practice browsers tend to set
* fairly long but finite time limits. If the time limit is exceeded
* for a request, the client will fail the request with an error. To
* prevent this, our main event loop (processNetRequests)
* periodically calls this routine if no other events have occurred
* recently. We'll clear out the pending event request queue for
* each client by sending a no-op reply to each event. This tells
* the client that the server is still alive and connected but has
* nothing new to report.
*/
sendKeepAlive()
{
/*
* Send no-op replies to any requests that have been in the queue
* for longer than the maximum event wait interval. New requests
* are added to the end of the queue, so the first item in the
* queue is the oldest.
*/
local t = getTime(GetTimeTicks);
while (pendingReqs.length() > 0 && t >= pendingReqs[1].reqTimeout)
{
/* pull out the oldest request */
local req = pendingReqs.shift();
try
{
/* send a no-op reply */
req.req.sendReply(
'<?xml version="1.0"?><event><keepAlive/></event>',
'text/xml', 200,
['Cache-control: no-store, no-cache, '
+ 'must-revalidate, post-check=0, pre-check=0',
'Pragma: no-cache',
'Expires: Mon, 26 Jul 1997 05:00:00 GMT']);
/* successful communications, so note the last up time */
updateEventTime();
}
catch (Exception exc)
{
/* couldn't send the reply - consider disconnecting */
checkDisconnect();
}
}
}
/* broadcast a downloadable file to all clients */
broadcastDownload(desc)
{
/* add the download to all clients */
webSession.clientSessions.forEach({c: c.addDownload(desc)});
}
/* add a download to this client */
addDownload(desc)
{
/* add the download to the table */
downloads[desc.resName] = desc;
}
/*
* Cancel a downloadable file. Removes the file from the download
* list and notifies the client that the file is no longer available.
*/
cancelDownload(desc)
{
/* remove the file from our table */
downloads.removeElement(desc.resName);
/* tell the client that the file is no longer available */
webMainWin.sendWinEventTo(
'<cancelDownload><<desc.resPath.htmlify()>></cancelDownload>',
self);
}
/* this client's list of downloadable temporary files */
downloads = perInstance(new LookupTable())
/* get a list of all of my downloadable files */
allDownloads() { return downloads.valsToList(); }
/* process the request and response queues */
processQueues()
{
/* keep going as long as we have requests and responses to pair up */
while (pendingReqs.length() > 0 && pendingEvts.length() > 0)
{
/* pull the first element out of each list */
local req = pendingReqs.shift();
local evt = pendingEvts[1];
/*
* Answer the request with the event. Since this is an API
* request masquerading as a page view, we need to generate a
* live reply to each new request to the same virtual
* resource path. So, include some headers to tell the
* browser and any proxies not to cache the reply.
*/
try
{
/* send the reply */
req.req.sendReply(
evt, 'text/xml', 200,
['Cache-control: no-store, no-cache, '
+ 'must-revalidate, post-check=0, pre-check=0',
'Pragma: no-cache',
'Expires: Mon, 26 Jul 1997 05:00:00 GMT']);
/* we've successfully sent the event - discard it */
pendingEvts.shift();
/* update the successful communications time */
updateEventTime();
}
catch (SocketDisconnectException sde)
{
/* the socket has been disconnected - consider disconnecting */
checkDisconnect();
}
catch (Exception exc)
{
/*
* Ignore other errors, since we can just turn around and
* send the event again in response to the next event
* request from the client - no information will be lost,
* since we still have the event in the queue.
*/
}
}
}
/* wait for the queues to empty in preparation for shutting down */
shutdownWait(timeout)
{
/* process network requests until all clients disconnect */
processNetRequests({: webSession.clientSessions.length() == 0 },
timeout);
}
/*
* Check to see if the client is still alive. If the client has no
* pending event requests, and we haven't heard from the client in
* more than the client session timeout interval, assume the client
* is no longer connected and kill the session object.
*
* This should be called whenever a sending a reply to a request
* fails with a Socket Disconnect exception. We also run this
* periodically during routine housekeeping to check for clients that
* haven't even bothered to send a request.
*/
checkDisconnect()
{
/* if we don't have any pending requests, kill the session */
if (pendingReqs.length() == 0
&& getTime(GetTimeTicks) >= lastEventTime + ClientSessionTimeout)
{
/* remove the client session from our master list */
webSession.removeClient(self);
/* the session is now dead */
isAlive = nil;
}
}
/*
* Class method: forcibly disconnect all clients. This simply
* deletes the list of active clients and deletes any pending events
* in their queues. This doesn't actually terminate their network
* connections, but simply clears out any pending work for each
* client that we've initiated on the server side.
*/
disconnectAll()
{
/* delete all pending events for each client */
webSession.clientSessions.forEach({ s: s.pendingReqs.clear() });
/* delete the client list */
webSession.clientSessions.clear();
}
/*
* Class method: Find a client session, given the session key or an
* HTTPRequest object.
*/
find(key)
{
/*
* if the key is given as an HTTPRequest, find the key by URL
* parameter or cookie
*/
if (dataType(key) == TypeObject && key.ofKind(HTTPRequest))
{
/* get the request */
local req = key;
/* try the URL parameters and cookies */
if ((key = req.getQueryParam('TADS_client')) == nil
&& (key = req.getCookie('TADS_client')) == nil)
{
/* there's no client key, so there's no client */
return nil;
}
}
/* find the client that matches the key string */
return webSession.clientSessions.valWhich({ x: x.clientKey == key });
}
/*
* Get the primary session. This is the session for the original
* initiating user (the "host" in a multi-user game).
*/
getPrimary()
{
/* scan for a session with the host session key */
return webSession.clientSessions.valWhich({ x: x.isPrimary });
}
;
/* ------------------------------------------------------------------------ */
/*
* Client event request. Each client session object keeps a queue of
* pending event requests, representing incoming "GET /webui/getEvent"
* requests that have yet to be answered.
*/
class ClientEventRequest: object
construct(req)
{
self.req = req;
reqTimeout = getTime(GetTimeTicks) + EventRequestTimeout;
}
/* the underlying HTTPRequest object */
req = nil
/*
* The system time (ms ticks) when the request times out. If we
* don't have an actual event to send in response by this time, the
* housekeeper will generate a no-op reply just to let the client
* know that we're still here.
*/
reqTimeout = 0
;
/* ------------------------------------------------------------------------ */
/*
* A WebResource is a virtual file accessible via the HTTP server. Each
* resource object has a path, which can be given as a simple string that
* must be matched exactly, or as a RexPattern object with a regular
* expression to be matched. Each object also has a "processRequest"
* method, which the server invokes to answer the request when the path
* is matched.
*/
class WebResource: object
/*
* The virtual path to the resource. This is the apparent URL path
* to this resource, as seen by the client.
*
* URL paths follow the Unix file system conventions in terms of
* format, but don't confuse the virtual path with an actual file
* system path. The vpath doesn't have anything to do with the disk
* file system on the server machine or anywhere else. That's why we
* call it "virtual" - it's merely the apparent location, from the
* client's perspective.
*
* When the server receives a request from the client, it looks at
* the URL sent by the client to determine which WebResource object
* should handle the request. The server does this by matching the
* resource path portion of the URL to the virtual path of each
* WebResource, until it finds a WebResource that matches. The
* resource path in the URL is the part of the URL following the
* domain, and continuing up to but not including any "?" query
* parameters. The resource path always starts with a slash "/".
* For example, for the URL "http://192.168.1.15/test/path?param=1",
* the resource path would be "/test/path".
*
* The virtual path can be given as a string or as a RexPattern. If
* it's a string, a URL resource path must match the virtual path
* exactly, including upper/lower case. If the virtual path is given
* as a RexPattern, the URL resource path will be matched to the
* pattern with the usual regular expression rules.
*/
vpath = ''
/*
* Process the request. This is invoked when we determine that this
* is the highest priority resource object matching the request.
* 'req' is the HTTPRequest object; 'query' is the parsed query data
* as returned by req.parseQuery(). The query information is
* provided for convenience, in case the result depends on the query
* parameters.
*/
processRequest(req, query)
{
/* by default, just send an empty HTML page */
req.sendReply('<html><title>TADS</title></html>', 'text/html', 200);
}
/*
* The priority of this resource. If the path is given as a regular
* expression, a given request might match more than one resource.
* In such cases, the matching resource with the highest priority is
* the one that's actually used to process the request.
*/
priority = 100
/*
* The group this resource is part of. This is the object that
* "contains" the resource, via its 'contents' property; any object
* will work here, since it's just a place to put the contents list
* for the resource group.
*
* By default, we put all resources into the mainWebGroup object.
*
* The point of the group is to allow different servers to use
* different sets of resources, or to allow one server to use
* different resource sets under different circumstances. When a
* server processes a request, it does so by looking through the
* 'contents' list for a group of its choice.
*/
group = mainWebGroup
/*
* Determine if this resource matches the given request. 'query' is
* the parsed query from the request, as returned by
* req.parseQuery(). 'req' is the HTTPRequest object representing
* the request; you can use this to extract more information from the
* request, such as cookies or the client's network address.
*
* This method returns true if the request matches this resource, nil
* if not.
*
* You can override this to specify more complex matching rules than
* you could achieve just by specifying the path string or
* RexPattern. For example, you could make the request conditional
* on the time of day, past request history, cookies in the request,
* parameters, etc.
*/
matchRequest(query, req)
{
/* get the query path */
local qpath = query[1];
/* by default, we match GET */
local verb = req.getVerb().toUpper();
if (verb != 'GET' && verb != 'POST')
return nil;
/* if the virtual path a string, simply match the string exactly */
if (dataType(vpath) == TypeSString)
return vpath == qpath;
/* if it's a regular expression, match the pattern */
if (dataType(vpath) == TypeObject && vpath.ofKind(RexPattern))
return rexMatch(vpath, qpath) != nil;
/* we can't match other path types */
return nil;
}
/*
* Send a generic request acknowledgment or reply. This wraps the
* given XML fragment in an XML document with the root type given by
* the last element in our path name. If the 'xml' value is omitted,
* we send "<ok/>" by default.
*/
sendAck(req, xml = '<ok/>')
{
/*
* Figure the XML document root element. If we have a non-empty
* path, use the last element of the path (as delimited by '/'
* characters). Otherwise, use a default root of <reply>.
*/
local root = 'reply';
if (dataType(vpath) == TypeSString
&& vpath.length() > 0
&& rexSearch(vpath, '/([^/]+)$') != nil)
root = rexGroup(1) [3];
/* send the reply, wrapping the fragment in a proper XML document */
sendXML(req, root, xml);
}
/*
* Send an XML reply. This wraps the given XML fragment in an XML
* document with the given root element.
*/
sendXML(req, root, xml)
{
req.sendReply('<?xml version="1.0"?>\<<<root>>><<xml>></<<root>>>',
'text/xml', 200);
}
;
/* ------------------------------------------------------------------------ */
/*
* A resource file request handler. This handles a request by sending
* the contents of the resource file matching the given name.
*
* To expose a bundled game resource as a Web object that the client can
* access and download via HTTP, simply create an instance of this class,
* and set the virtual path (the vpath property) to the resource name.
* See coverArtResource below for an example - that object creates a URL
* for the Cover Art image so that the browser can download and display
* it.
*
* You can expose *all* bundled resources in the entire game simply by
* creating an object like this:
*
*. WebResourceResFile
*. vpath = static new RexPattern('/')
*. ;
*
* That creates a URL mapping that matches *any* URL path that
* corresponds to a bundled resource name. The library intentionally
* doesn't provide an object like this by default, as a security measure;
* the default configuration as a rule tries to err on the side of
* caution, and in this case the cautious thing to do is to hide
* everything by default. There's really no system-level security risk
* in exposing all resources, since the only files available as resources
* are files you explicitly bundle into the build anyway; but even so,
* some resources might be for internal use within the game, so we don't
* want to just assume that everything should be downloadable.
*
* You can also expose resources on a directory-by-directory basis,
* simply by specifying a longer path prefix:
*
*. WebResourceResFile
*. vpath = static new RexPattern('/graphics/')
*. ;
*
* Again, the library doesn't define anything like this by default, since
* we don't want to impose any assumptions about how your resources are
* organized.
*/
class WebResourceResFile: WebResource
/*
* Match a request. A resource file resource matches if we match the
* virtual path setting for the resource, and the requested resource
* file exists.
*/
matchRequest(query, req)
{
return inherited(query, req) && resExists(processName(query[1]));
}
/* process the request: send the resource file's contents */
processRequest(req, query)
{
/* get the local resource name */
local name = processName(query[1]);
/* get the filename suffix (extension) */
local ext = nil;
if (rexSearch('%.([^.]+)$', name) != nil)
ext = rexGroup(1) [3];
local fp = nil;
try
{
/* open the file in the appropriate mode */
if (isTextFile(name))
fp = File.openTextResource(name);
else
fp = File.openRawResource(name);
}
catch (FileException exc)
{
/* send a 404 error */
req.sendReply(404);
return;
}
/*
* If the file suffix implies a particular mime type, set it.
* There are some media types that are significant to browsers,
* but which the HTTPRequest object can't infer based on the
* contents, so as a fallback infer the media type from the
* filename suffix if possible.
*/
local mimeType = browserExtToMime[ext];
/*
* Send the file's contents. Since resource files can be large
* (e.g., images or audio files), send the reply asynchronously
* so that we don't block other requests while the file is being
* downloaded. Browsers typically download multimedia resources
* in background threads so that the UI remains responsive during
* large downloads, so we have to be prepared to handle these
* overlapped requests while a download proceeds.
*/
req.sendReplyAsync(fp, mimeType);
/* done with the file */
fp.closeFile();
}
/* extension to MIME type map for important browser file types */
browserExtToMime = static [
'css' -> 'text/css',
'js' -> 'text/javascript'
]
/*
* Process the name. This takes the path string from the query, and
* returns the resource file name to look for. By default, we simply
* return the same name specified by the client, minus the leading
* '/' (since resource paths are always relative).
*/
processName(n) { return n.substr(2); }
/*
* Determine if the given file is a text file or a binary file. By
* default, we base the determination solely on the filename suffix,
* checking the extension against a list of common file types.
*/
isTextFile(fname)
{
/* get the extension */
if (rexMatch('.*%.([^.]+)$', fname) != nil)
{
/* pull out the extension */
local ext = rexGroup(1) [3].toLower();
/*
* check against common binary types - if it's not there,
* assume it's text
*/
return (binaryExts.indexOf(ext) == nil);
}
else
{
/* no extension - assume binary */
return nil;
}
}
/* table of common binary file extensions */
binaryExts = ['jpg', 'jpeg', 'png', 'mng', 'bmp', 'gif',
'mpg', 'mp3', 'mid', 'ogg', 'wav',
'pdf', 'doc', 'docx', 'swf',
'dat', 'sav', 'bin', 'gam', 't3', 't3v'];
;
/*
* The resource handler for our standard library resources. All of the
* library resources are in the /webuires resource folder. This exposes
* everything in that folder as a downloadable Web object.
*/
webuiResources: WebResourceResFile
vpath = static new RexPattern('/webuires/')
;
/* the special cover art resource */
coverArtResource: WebResourceResFile
vpath = static new RexPattern('/%.system/CoverArt%.(jpg|png)$')
;
/*
* Session initializer resource. This is a mix-in class designed to be
* used for a special resource that initializes the session. Mix this
* with a WebResource class to set up the initializer. When you connect
* to the client via connectWebUI(), point the client to this resource.
*
* There are two elements to setting up the session. First, we need to
* set the program session key as a cookie. The client obtains this from
* the registration mechanism, whose purpose is to launch the game
* program and send the connection information back to the client. The
* client sends this to us in the form of a URL parameter, TADS_session.
* This key is essentially for authentication, to make sure that the
* client that we're talking to is actually the client that launched the
* program: only that client would be able to get the key, because we
* invent it and send it to the registrar, and the registrar only sends
* it back to the client session it's already established on its end.
* This prevents port scanners from finding our open port and trying to
* crawl our "site" or otherwise access our services.
*
* The second setup element is to create a client session key. Whereas
* the program session key is for our entire service, the client session
* key is specific to this one connection. If the user opens two browser
* windows on this server, each browser needs its own separate client
* session so that we can tell the traffic apart. The client session key
* is simply another random key we generate, and again we pass it back to
* the client in a cookie.
*
* The reason we set cookies for both of these session keys is that it
* lets the client pass the information back to us on subsequent requests
* without having to encode another parameter in every URL. We set
* session cookies in both cases; the program session is for
* authentication purposes and so we don't want it to be stored or
* shared, and the client session key is explicitly to identify this one
* browser session, so it obviously shouldn't be shared across browser
* instances or sessions.
*
* Note that instances should always provide a string (as opposed to a
* regular expression) for the virtual path (the 'vpath' property). We
* have to send the path to the browser UI as part of the connection
* information, so we need a string we can send rather than a pattern to
* match.
*/
class WebResourceInit: object
/*
* Connect to the client. The program should call this after
* creating its HTTPServer object, which you pass here as 'srv'.
* This establishes the client UI connection, generating the path to
* the start page.
*/
connectUI(srv)
{
/*
* point the client to our start page, adding the session key as
* a query parameter
*/
connectWebUI(srv, '<<vpath>>?TADS_session=<<webSession.sessionKey>>');
}
/*
* Process the request. This sets up the program and client session
* keys as cookies.
*/
processRequest(req, query)
{
/* get the session parameter from the query */
local skey = query['TADS_session'];
/* set the session cookie in the reply */
if (skey)
req.setCookie('TADS_session', '<<skey>>; path=/;');
/* check for a client session */
local ckey = req.getCookie('TADS_client');
if (ckey == nil || ClientSession.find(ckey) == nil)
{
/*
* There's no client session ID cookie, or the ID is invalid
* or has expired. Create a new client session.
*/
/* get the storage server session ID key from the request */
local ssid = query['storagesid'];
/* get the user name */
local uname = query['username'];
/*
* if there's no storage server session, and they're
* connecting under the primary server session key, use the
* primary user's storage server session
*/
if (ssid == nil && skey == webSession.sessionKey)
ssid = webSession.storageSID;
/* create the new session object */
local client = new transient ClientSession(skey, ssid);
/* if there's a user name, set it in the session */
if (uname != nil)
client.screenName = uname;
else
client.setDefaultScreenName();
/* send the client session ID to the browser as a cookie */
req.setCookie('TADS_client', '<<client.clientKey>>; path=/;');
/* if this is a guest, alert everyone to the new connection */
if (!client.isPrimary)
webMainWin.postSyntheticEvent('newGuest', client.screenName);
}
/* go return the underlying resource */
inherited(req, query);
}
/* the HTPTServer for communicating with the client */
server = nil
;
/* ------------------------------------------------------------------------ */
/*
* A WebResourceGroup is a container for WebResource objects. When a
* server receives a request, it looks in its group list to find the
* resource object that will handle the request.
*/
class WebResourceGroup: object
/*
* Should this group handle the given request? By default, we say
* yes if the server that received the request is associated with
* this group via the group's 'server' property.
*/
isGroupFor(req)
{
/* get the request's server object */
local srv = req.getServer();
/*
* if this server matches our 'server' property, or is in the
* list of servers if 'server' is a list, we're the group for the
* request
*/
return (srv == server
|| (dataType(server) == TypeList
&& server.indexOf(srv) != nil));
}
/*
* The priority of the group, relative to other groups. If the same
* server matches multiple groups, this allows you to designate which
* group has precedence. A higher value means higher priority.
*/
priority = 100
/*
* The HTTPServer object or objects this group is associated with.
* The general event processor uses this to route a request to the
* appropriate resource group, by finding the group that's associated
* with the server that received the request.
*
* To associate a group with multiple servers, make this a list.
*/
server = nil
/* the WebResource objects in the group */
contents = []
/*
* Process a request. This looks for the highest priority matching
* resource in the group, then hands the request to that resource for
* processing.
*/
processRequest(req)
{
/* parse the query */
local query = req.parseQuery();
/*
* Check for the session ID. The session ID is required either
* in a URL parameter or in a cookie. If it's not present,
* reject the request with a "403 Forbidden" error, since the
* session is essentially an authentication token to tell us that
* the client is in fact the same user that launched the game.
*/
if (!webSession.validateKey(req, query))
return;
/*
* Search our list for the first resource that matches this
* request. The list is initialized in descending priority
* order, so the first match we find will be the one with the
* highest priority.
*/
local match = contents.valWhich({res: res.matchRequest(query, req)});
/* if we found a match, process it; otherwise return a 404 error */
if (match != nil)
match.processRequest(req, query);
else
req.sendReply(404);
}
/* class property: list of all WebResourceGroup objects */
all = []
;
/* ------------------------------------------------------------------------ */
/*
* The default web resource group. This is the default container for
* WebResource objects.
*/
mainWebGroup: WebResourceGroup
/* the default group matches any server, but with low priority */
isGroupFor(req) { return true; }
priority = 1
;
/* ------------------------------------------------------------------------ */
/*
* At startup, put each WebResource object into the contents list for its
* group.
*/
PreinitObject
execute()
{
/* build the contents list for each resource group */
forEachInstance(WebResource, function(obj) {
/* get the group object for the resource */
local g = obj.group;
/* if the group doesn't have a contents list yet, create one */
if (g.contents == nil)
g.contents = [];
/* add this resource to the contents list */
g.contents += obj;
});
/* sort each group's contents list in priority order */
forEachInstance(WebResourceGroup, function(obj) {
/* sort the group's contents list */
obj.contents = obj.contents.sort(
SortDesc, { a, b: a.priority - b.priority });
/* add this group to the master list of groups */
WebResourceGroup.all += obj;
});
/* sort the groups in descending order of priority */
WebResourceGroup.all = WebResourceGroup.all.sort(
SortDesc, { a, b: a.priority - b.priority });
}
;
/* ------------------------------------------------------------------------ */
/*
* Guest connection request. This enables "switchboard" applications on
* remote servers that keep track of multi-user game sessions, to show
* users available sessions and connect new users to those sessions.
*
* The first step in setting up a switchboard is for the game server to
* register itself with the switchboard by sending a request on startup.
* That part is external to us - that's not handled within the game
* program but rather within the web server script that launches the
* game. Here, then, we simply assume that this work is already done.
*
* The second step is that the switchboard needs to check back with the
* game server from time to time to see if it's still alive - essentially
* a "ping" operation. We handle that here: if we respond to the
* request, we're obviously still alive.
*
* The third step is that we need to send the switchboard a URL that lets
* secondary users ("guests") connect to the game session. We handle
* that here as well: our reply body is the client connection URL.
*/
guestConnectPage: WebResource
vpath = '/webui/guestConnect'
processRequest(req, query)
{
/* send the collaborative connection URL */
req.sendReply(webSession.getCollabUrl(), 'text/plain', 200);
}
;
/* ------------------------------------------------------------------------ */
/*
* getEvent request. This is the mechanism we use to "send" events to
* the client. The client sends a getEvent request to us, and we simply
* put it in a queue - we don't send back any response immediately. As
* soon as we want to send an event to the client, we go through the
* queue of pending getEvent requests, and reply to each one with the
* event we want to send.
*/
eventPage: WebResource
vpath = '/webui/getEvent'
processRequest(req, query)
{
/* find the client */
local c = ClientSession.find(req);
/* if we found the client session object, send it the request */
if (c != nil)
c.requestEvent(req);
else
req.sendReply(400);
}
/* broadcast an event message to each client */
sendEvent(msg)
{
/* build the full XML message */
msg = '<?xml version="1.0"?><event><<msg>></event>';
/* send it to each client */
ClientSession.broadcastEvent(msg);
}
/* send an event to a particular client */
sendEventTo(msg, client)
{
/* build the full XML message */
msg = '<?xml version="1.0"?><event><<msg>></event>';
/* send it to the given client */
client.sendEvent(msg);
}
;
/*
* Flush events. This cancels any pending event requests for the client.
* The client can use this after being reloaded to flush any outstanding
* event requests from a past incarnation of the page.
*/
flushEventsPage: WebResource
vpath = '/webui/flushEvents'
processRequest(req, query)
{
/* find the client */
local c = ClientSession.find(req);
/* if we found the client, send it the flush request */
if (c != nil)
c.flushEvents();
/*
* acknowledge the request; it's okay if we didn't find the
* client session, since this just means there are no events to
* flush for this client
*/
sendAck(req);
}
;
/*
* getState request. The web page can send this to get a full accounting
* of the current state of the UI. It does this automatically when first
* loaded, and again when the user manually refreshes the page.
*
* We handle this by asking the main window to generate its state.
*/
uiStatePage: WebResource
vpath = '/webui/getState'
processRequest(req, query)
{
/* get the window making the request */
local w = webMainWin.winFromPath(query['window']);
local client = ClientSession.find(req);
/* if we found the window, send the reply */
if (w)
{
/* send the uiState reply for the window */
sendXML(req, 'uiState', w.getState(client));
}
else
{
/* no window - send an error reply */
req.sendReply(406);
}
}
;
/* ------------------------------------------------------------------------ */
/*
* Process network requests. Continues until doneFunc() returns true, or
* a timeout or error occurs. If we return because doneFunc() returned
* true, we'll return nil. Otherwise, we'll return the NetEvent that
* terminated the wait.
*/
processNetRequests(doneFunc, timeout?)
{
/* if there's a timeout, figure the ending time */
local endTime = (timeout != nil ? getTime(GetTimeTicks) + timeout : nil);
/* keep going until the 'done' function returns true */
while (doneFunc == nil || !doneFunc())
{
try
{
/* get the next housekeeping time */
local hkTime = webSession.hkTime;
/*
* figure the time to the next timeout - stop at the caller's
* ending time, or the next housekeeping time, whichever is
* sooner
*/
local tf = (endTime != nil && endTime < hkTime ? endTime : hkTime);
/* figure the time remaining to the next timeout */
local dt = max(0, tf - getTime(GetTimeTicks));
/*
* Flush any pending output. If we're waiting for user
* input, this ensures that the last output is visible in the
* UI while we await the next user action.
*/
flushOutput();
/* get the next network event */
local evt = getNetEvent(dt);
/* see what we have */
switch (evt.evType)
{
case NetEvRequest:
/*
* This is an incoming network request from a client.
* Check the protocol type of the request object.
*/
if (evt.evRequest.ofKind(HTTPRequest))
{
/*
* HTTP request - process it through the appropriate
* web resource group. First, find the group that
* wants to handle the request.
*/
local req = evt.evRequest;
local group = WebResourceGroup.all.valWhich(
{ g: g.isGroupFor(req) });
/*
* if there's a client associated with the request,
* update its last activity time
*/
local client = ClientSession.find(req);
if (client != nil)
client.updateEventTime();
/* if we found a group, let it handle the request */
if (group != nil)
{
try
{
/* send the request to the group for processing */
group.processRequest(req);
}
catch (Exception exc)
{
/*
* Unhandled exception - something went wrong
* in the server, so the appropriate reply is
* 500 Internal Server Error. Send the
* exception message with the reply.
*/
local msg =
'Unhandled exception processing request: '
+ exc.getExceptionMessage().specialsToText();
req.sendReply(msg, 'text/plain', 500);
}
}
else
{
/* no group - send back a 404 error */
req.sendReply(404);
}
}
break;
case NetEvTimeout:
/*
* Timeout. Always try running housekeeping on a
* timeout; the housekeeper will ignore this unless the
* time has actually come.
*/
hkTime = webSession.housekeeping();
/*
* Check for a caller timeout. If the caller's end time
* has arrived, pass the timeout back to the caller as a
* timeout event.
*/
if (endTime != nil && getTime(GetTimeTicks) >= endTime)
return evt;
/*
* it wasn't a caller timeout, so it must have been an
* internal housekeeping timeout, which we just handled;
* simply continue looping
*/
break;
case NetEvUIClose:
/*
* UI Closed. This tells us that the user has manually
* closed the UI window. This only happens in the local
* stand-alone configuration, where the "browser" is an
* integrated part of the interpreter application,
* simulating the traditional TADS interpreter setup
* where the whole program is a single application
* running on one machine. In this setup, closing the UI
* window should dismiss the whole application, since
* that's the convention on virtually all GUIs.
*/
/* disconnect all clients */
ClientSession.disconnectAll();
/* quit by signaling a "quit game" event */
throw new QuittingException();
}
}
catch (SocketDisconnectException sdx)
{
/* the client has closed its connection; ignore this */
}
}
/* indicate that the 'done' condition was triggered */
return nil;
}
/* ------------------------------------------------------------------------ */
/*
* Web Window tracker. This is a game object that controls and remembers
* the state of a "window" in the browser user interface. By "window",
* we basically mean an HTML page, which might reside at the top level of
* the browser itself, or inside an IFRAME element within an enclosing
* page.
*
* Each WebWindow class corresponds to a particular HTML page that we
* serve the client. The HTML page is the expression of the window in
* the browser, and the WebWindow object is the expression of the same
* information in the game program. The two are different facets of the
* same conceptual UI object. The reason we need the two separate
* expressions is that the server controls everything, but the client has
* to do the actual display work, and the two parts of the program speak
* different languages - the server is TADS, and the client is HTML.
*
* The WebWindow object on the server lets us easily reconstruct the UI
* state in a newly opened browser window, or when the user performs a
* page refresh. This object's job is to send information to the client
* on demand that allows the client to display the page in its current
* state.
*
* Note that a given WebWindow/HTML page combination can be used more
* than once within the same UI. The pages defined in the library are
* designed to be generic and reusable, so you might use the same window
* class more than once for different purposes within the UI. The
* library pages can also be subclassed, by subclassing the WebWindow
* object and creating a customized copy of the corresponding HTML page
* resource.
*/
class WebWindow: WebResourceResFile
/*
* The URL path to the window's HTML definition file, as seen by the
* browser. For the pre-defined library window types, we expose the
* HTML file in the root of the URL namespace - e.g., "/main.html".
* The files are actually stored in the /webuires folder, but we
* expose them to the browser as though they were in the root folder
* to make embedded object references on the pages simpler. The
* browser figures the path to an embedded object relative to the
* containing page, so by placing the containing page in the root
* folder, embedded object paths don't have to worry about
* referencing parent folders.
*/
vpath = nil
/*
* The window's actual source location, as a resource path. A given
* WebWindow subclass corresponds to a particular HMTL page, since
* the class and the page are facets of the same conceptual object
* (one facet is the browser expression, the other is the game
* program expression).
*/
src = nil
/* process a request path referencing me into my actual resource path */
processName(n) { return src; }
/*
* Resolve a window path name. For container windows, this should
* search the sub-windows for the given path. By default, we match
* simply if the path matches our name.
*/
winFromPath(path)
{
return path == name ? self : nil;
}
/*
* Flush the window. This sends any buffered text to the UI.
*/
flushWin() { }
/*
* Write text to the window. Subclasses with stream-oriented APIs
* must override this.
*/
write(txt) { }
/*
* Clear the window. Subclasses must override this.
*/
clearWindow() { }
/*
* Get the window's current state. This returns a string containing
* an XML fragment that describes the state of the window. This
* information is sent to the HTML page when the browser asks for the
* current layout state when first loaded or when the page is
* refreshed. The XML format for each subclass is specific to the
* Javascript on the class's HTML page.
*/
getState(client) { return ''; }
/* send an event related to this window to all clients */
sendWinEvent(evt)
{
/*
* send the event message, adding a <window> parameter to
* identify the source
*/
eventPage.sendEvent('<window><<pathName>></window><<evt>>');
}
/* send a window event to a specific client */
sendWinEventTo(evt, client)
{
eventPage.sendEventTo('<window><<pathName>></window><<evt>>', client);
}
/* specialsToHtml context */
sthCtx = perInstance(new SpecialsToHtmlState())
/* the name of this window */
name = nil
/* the full path name of this window, in "win.sub.sub" format */
pathName = nil
;
/* ------------------------------------------------------------------------ */
/*
* Layout Window. This is a specialized Web Window tracker for our
* layout page type, which is displayed using the resource file
* webuires/layout.html. This page is designed as a container of more
* specialized sub-window pages; its job is to divide up the window space
* into IFRAME elements that display the sub-windows, and to manage the
* geometry of the IFRAMEs.
*
* The layout page is primarily designed to be the top-level page of the
* web UI. The idea is to set up a layout page as the navigation URL for
* the browser, so the layout page fills the browser window. You then
* arrange your functional windows within the layout page - a command
* window, a status line window, etc. This arrangement is similar to
* banner window in HTML TADS, but IFRAMEs are considerably more
* flexible; for example, they don't have to tile the main window, and
* you can size them in the full range of units CSS provides.
*
* Layout windows aren't limited to the top level, though. Since you can
* put any HTML page within an IFRAME, you can put another layout window
* within an IFRAME, to further subdivide the space inside the IFRAME.
*/
class WebLayoutWindow: WebWindow
/*
* Resolve a window path name
*/
winFromPath(path)
{
/* get the first element and the rest of the path */
local idx = path.find('.');
if (idx == nil)
idx = path.length() + 1;
/* pull out the first element and the rest */
local head = path.substr(1, idx - 1);
local tail = path.substr(idx + 1);
/* if the first element doesn't match our name, it's not a match */
if (head != name)
return nil;
/* if that's the end of the path, we have our match */
if (tail == '')
return self;
/* match the rest of the path against our children */
foreach (local w in frames)
{
local match = w[1].winFromPath(tail);
if (match != nil)
return match;
}
/* no match */
return nil;
}
/*
* Create a new window within the layout. This creates an IFRAME in
* the browser, laid out according to the 'pos' argument, and
* displays the given window object within the frame.
*
* If the window already exists, this updates the window with the new
* layout settings.
*
* 'win' is a WebWindow object that will be displayed within the
* IFRAME. This method automatically loads the HTML resource from
* the WebWindow into the new IFRAME.
*
* 'name' is the name of the window. Each window within a layout
* must have a distinct name. This allows you to refer to the
* dimensions of other windows in 'pos' parameters. The name should
* be alphanumeric.
*
* 'pos' is the layout position for the new frame. This is a string
* in this format: 'left, top, width, height', where 'left' is the
* horizontal position of the top left corner, 'top' is the vertical
* position of the top left corner, 'width' is the width of the
* window, and 'height' is the height. Each element can be specified
* as a Javascript-style arithmetic expression. Within the
* expression, you can use a mix of any of the following:
*
*. 123 - a number, representing a number of pixels on the display
*. 5em - 5 'em' units, relative to the main BODY font in the window
*. 5en - 5 'en' units in the main BODY font
*. 5ex - 5 'ex' units in the main BODY font
*. window.width - the width in pixels of the enclosing window
*. window.height - the height in pixels of the enclosing window
*. 50% - percentage of the width or height of the enclosing window
*. content.width - the width in pixels of the contents of the frame
*. content.height - the height in pixels of the contents of the frame
*. x.left - horizontal coordinate of leftmost edge of window 'x'
*. x.right - horizontal coordinate of rightmost edge of window 'x'
*. x.top - vertical coordinate of top edge of window 'x'
*. x.bottom - vertical coordinate of bottom edge of window 'x'
*. x.width - width in pixels of window 'x'
*. x.height - height in pixels of window 'x'
*
* The "window" dimensions refer to the *enclosing* window. If this
* layout window is the main page of the UI, this is simply the
* browser window itself. For a layout window nested within another
* frame, this is the enclosing frame.
*
* Percentage units apply to the enclosing window. When a percentage
* is used in the 'left' or 'width' slot, it applies to the width of
* the enclosing window; in the 'top' or 'height' slot, it applies to
* the height.
*
* The "content" dimensions refer to the contents of the frame we're
* creating. This is the size of the contents as actually laid out
* in the browser.
*
* "x.left" and so on refer to the dimensions of other frames *within
* this same layout window*. 'x' is the name of another window
* within the same layout, as specified by the 'name' argument given
* when the window was created.
*/
createFrame(win, name, pos)
{
/* set the window's internal name to its full path name */
win.name = name;
win.pathName = self.name + '.' + name;
/* add the window to our list */
frames[name] = [win, pos];
/* notify the UI */
sendWinEvent('<subwin>'
+ '<name><<name>></name>'
+ '<pos><<pos>></pos>'
+ '<src><<win.vpath.htmlify()>></src>'
+ '</subwin>');
}
/*
* Flush this window. For a layout window, we simply flush each
* child window.
*/
flushWin()
{
/* flush each child window */
frames.forEach({w: w[1].flushWin()});
}
/*
* Get the state.
*/
getState(client)
{
/* build an XML fragment describing the list of frames */
local s = '';
frames.forEachAssoc(function(name, info) {
s += '<subwin><name><<name>></name>'
+ '<pos><<info[2]>></pos>'
+ '<src><<info[1].vpath.htmlify()>></src>'
+ '</subwin>';
});
/* return the completed state object */
return s;
}
/*
* The table of active frames within this layout. This table is
* keyed by window name; each entry is a list of [win, pos], where
* 'win' is the WebWindow object for the window, and 'pos' is its
* position parameter.
*/
frames = perInstance(new LookupTable(16, 32))
/* my virtual path and the actual resource file location */
vpath = '/layoutwin.html'
src = 'webuires/layoutwin.html'
;
/* ------------------------------------------------------------------------ */
/*
* Command Window. This object keeps track of the state of command
* window within the web UI.
*/
class WebCommandWin: WebWindow
/*
* Write to the window
*/
write(txt)
{
/* add the text to the output buffer */
outbuf.append(txt);
}
/*
* Flush the buffers
*/
flushWin()
{
/* get the current output buffer */
local txt = toString(outbuf).specialsToHtml(sthCtx);
/* add it to the state buffer (text since last input) */
textbuf.append(txt);
/* send the text to the client via an event */
sendWinEvent('<say><<txt.htmlify()>></say>');
/* we've now processed the pending output buffer */
outbuf.deleteChars(1);
}
/*
* Read a line of input in this window. Blocks until the reply is
* received. Returns nil on timeout.
*/
getInputLine(timeout?)
{
/* flush buffered output */
flushWin();
/* clear out the last input */
lastInput = nil;
lastInputClient = nil;
lastInputReady = nil;
/* set the UI state */
mode = 'inputLine';
isInputOpen = true;
/* send the inputLine event to the client */
sendWinEvent('<inputLine/>');
/* process network events until we get our input or we time out */
processNetRequests(
{: lastInputReady || webMainWin.syntheticEventReady() },
timeout);
/* move the current textbuf contents to the scrollback list */
textbufToScrollback(lastInput);
/* back to 'working' mode */
mode = 'working';
/* check the result */
if (lastInput != nil)
{
/* we got a reply - mark the input line as closed in the UI */
isInputOpen = nil;
/* reset to the start of the line in the output context */
sthCtx.resetLine();
/* remember the source of the command */
webMainWin.curCmdClient = lastInputClient;
/* return the Line Input event */
return [InEvtLine, lastInput];
}
else if (webMainWin.syntheticEventReady())
{
/* there's a synthetic event available - return it */
return webMainWin.getSyntheticEvent();
}
else
{
/* we didn't get a reply, so we timed out */
return [InEvtTimeout, ''];
}
}
/*
* Cancel an input line that was interrupted by a timeout
*/
cancelInputLine(reset)
{
/* if the input line is open, send the cancel event */
if (isInputOpen)
{
/* send the cancel event to the client */
sendWinEvent('<cancelInputLine reset="<<
reset ? 'yes' : 'no'>>" />');
/* the input line is closed */
isInputOpen = nil;
/* reset to the start of the line in the output context */
sthCtx.resetLine();
}
}
/*
* Get the state of this command window
*/
getState(client)
{
return '<mode><<mode>></mode><scrollback>'
+ scrollback.join()
+ '<sbitem><text><<toString(textbuf).htmlify()>></text></sbitem>'
+ '</scrollback>';
}
/*
* Receive input from the client
*/
receiveInput(req, query)
{
/* remember the text */
lastInput = query['txt'];
/* remember the source of the input */
lastInputClient = ClientSession.find(req);
/* set the input-ready flag so we exit the modal input loop */
lastInputReady = true;
/* get the user who entered the command */
local user = (lastInputClient != nil
? lastInputClient.screenName : '');
/* tell any other windows listening in about the new input */
sendWinEvent('<closeInputLine>'
+ (lastInput != nil ?
'<text><<lastInput.htmlify()>></text>' : '')
+ '<user><<user.htmlify()>></user>'
+ '</closeInputLine>');
/* reset to the start of the line in the output context */
sthCtx.resetLine();
}
/*
* Clear the window
*/
clearWindow()
{
/* flush the output buffer */
flushWin();
/*
* clear the transcript - if the user refreshes the browser
* window, we want to show a blank window
*/
textbuf.deleteChars(1);
scrollback.clear();
/* send a clear window event */
sendWinEvent('<clearWindow/>');
/* reset to the start of the line in the output context */
sthCtx.resetLine();
}
/*
* Move the current text buffer contents to the scrollback list. If
* this would make the scrollback list exceed the limit, we'll drop
* the oldest item.
*
* 'cmd' is the command line text of the last input. We include this
* in the srollback list with special tagging so that the UI can
* display it in a custom style, if it wants.
*/
textbufToScrollback(cmd)
{
/* get the current buffer, tagged as a <text> entry in the list */
local t = '<text><<toString(textbuf).htmlify()>></text>';
/* add the command line tagged as <input> */
if (cmd != nil)
t += '<input><<cmd.htmlify()>></input>';
/* add the buffer to the scrollback list */
scrollback.append('<sbitem><<t>></sbitem>');
/* if this pushes us past the limit, drop the oldest item */
if (scrollback.length() > scrollbackLimit)
scrollback.shift();
/* clear text buffer */
textbuf.deleteChars(1);
}
/*
* Show a "More" prompt
*/
showMorePrompt()
{
/* flush the output buffer */
flushWin();
/* send a "More" prompt event to the UI */
sendWinEvent('<morePrompt/>');
mode = 'morePrompt';
moreMode = true;
/* process events until More mode is done */
processNetRequests({: !moreMode });
/* return the default mode */
mode = 'working';
}
/*
* receive notification from the client that the user has responded
* to the More prompt, ending the pause
*/
endMoreMode()
{
/* no longer in More mode */
moreMode = nil;
}
/* main window text buffer since last input read */
textbuf = perInstance(new StringBuffer(4096))
/*
* Scrollback list. After each input, we add the contents of
* 'textbuf' to this list. If this pushes the list past the limit,
* we drop the oldest item. This is used to reconstruct a reasonable
* amount of scrollback history when a new client connects, or when
* an existing client refreshes the page.
*/
scrollback = perInstance(new Vector())
/*
* The scrollback limit, as a number of command inputs. Each input
* interaction adds one item to the scrollback list. When the number
* of items in the list exceeds the limit set here, we drop the
* oldest item.
*/
scrollbackLimit = 10
/* pending output buffer, since last flush */
outbuf = perInstance(new StringBuffer(4096))
/* the text of the last input line we received from the client */
lastInput = nil
/* is input ready? */
lastInputReady = nil
/* client session who sent the last input line */
lastInputClient = nil
/*
* Is an input line open? This is true between sending an
* <inputLine> event and either getting a reply, or explicitly
* sending a close or cancel event.
*/
isInputOpen = nil
/* flag: we're in More mode */
moreMode = nil
/*
* Current UI mode. This is 'working' if the program is running and
* in the process of computing and/or generating output; 'inputLine'
* if we're waiting for the user to enter a line of input;
* 'morePrompt' if we're showing a "More" prompt.
*/
mode = 'working'
/* my virtual path, and the actual resource file location */
vpath = '/cmdwin.html'
src = 'webuires/cmdwin.html'
;
/*
* input-line event page
*/
inputLinePage: WebResource
vpath = '/webui/inputLine'
processRequest(req, query)
{
/* find the window */
local w = webMainWin.winFromPath(query['window']);
/* dispatch to the window if we found it */
if (w != nil)
{
/* send the input to the window */
w.receiveInput(req, query);
/* acknowledge the request */
sendAck(req);
}
else
req.sendReply(406);
}
;
/*
* "More" prompt done event page
*/
morePromptDonePage: WebResource
vpath = '/webui/morePromptDone'
processRequest(req, query)
{
/* find the target winodw */
local w = webMainWin.winFromPath(query['window']);
if (w != nil)
{
/* release More mode in the server window */
w.endMoreMode();
/* acknowledge the request */
sendAck(req);
}
else
req.sendReply(406);
}
;
/* ------------------------------------------------------------------------ */
/*
* Set Preferences command
*/
setPrefsPage: WebResource
vpath = '/webui/setPrefs'
processRequest(req, query)
{
/* get the request body */
local f = req.getBody();
if (f == nil)
{
errorReply(req, 'Error saving preferences: no data received');
return;
}
else if (f == 'overflow')
{
errorReply(req, 'Error saving preferences: message too large');
return;
}
/* get the client session for the request */
local cli = ClientSession.find(req);
if (cli == nil)
{
errorReply(req, 'Error saving preferences: missing session');
return;
}
/* process the file through the profile reader */
cli.uiPrefs.readSettings(f);
/* save the updated settings */
cli.uiPrefs.saveSettings();
/* done with the file */
f.closeFile();
/* done - acknowledge the request */
sendAck(req);
}
/* send an error as the reply to a request, formatted into XML */
errorReply(req, msg)
{
sendXML(req, 'reply', '<error><<msg.htmlify()>></error>');
}
;
/*
* UI Settings list. This represents a named UI settings profile in the
* Web UI. A profile is a list of name/value pairs.
*
* Most of the name keys are style IDs defined in the javascript for on
* the UI side - see main.js. These style IDs are arbitrary keys we
* define to identify UI elements - "mainFont" for the main font name,
* "statusBkg" for the status-line window's background color, etc. Each
* style ID generally corresponds to a dialog control widget in the
* preferences dialog in the javascript UI, and also corresponds to one
* or more CSS style selectors. The mapping from style ID to CSS is
* defined in the UI javascript (see prefsMapper in main.js).
*
* The non-style key "profileName" is the user-visible name of this
* profile. Internally, we refer to profiles using ID values, which are
* arbitrary identifiers generated by the UI when it creates a new
* profile (it currently uses integer keys).
*/
class WebUIProfile: object
construct(id)
{
self.profileID = id;
self.settings = new LookupTable();
}
/* set a preference item in the profile */
setItem(id, val)
{
settings[id] = val;
}
/* call a callback for each style: func(id, val) */
forEach(func)
{
settings.forEachAssoc(func);
}
/* internal ID of the profile */
profileID = ''
/* table of style value strings, keyed by style ID */
settings = nil
;
/*
* Web UI preferences. This object contains the in-memory version of the
* display style preferences file.
*
* Each client session has its own copy of this object, because each
* client can be associated with a different user, and each user has
* their own preferences file.
*/
class WebUIPrefs: object
construct(c)
{
/* remember our client session object */
clientSession = c;
/* load the initial settings from the user's config file */
loadSettings();
}
/* read the settings file */
loadSettings()
{
/* open the preferences file; do nothing if that fails */
local f = openSettingsFile(FileAccessRead);
if (f == nil)
return;
/* read the settings from the file */
readSettings(f);
/* done with the file */
f.closeFile();
}
/* read settings from a file */
readSettings(f)
{
/* set up our table of profiles */
local pros = profileTab = new LookupTable();
/* we don't have a current profile selection yet */
curProfile = nil;
/* read the file */
for (;;)
{
/* read the next line */
local l = f.readFile();
if (l == nil)
break;
/* if it's a valid line, process it */
if (rexMatch(curProPat, l) != nil)
{
/* current profile setting */
curProfile = rexGroup(1) [3];
}
else if (rexMatch(proItemPat, l) != nil)
{
/* style item definition - pull out the parts */
local proid = rexGroup(1) [3];
local key = rexGroup(2) [3];
local val = rexGroup(3) [3];
/* if the profile isn't in the table yet, add it */
local pro = pros[proid];
if (pro == nil)
pros[proid] = pro = new WebUIProfile(proid);
/* add this item to the table */
pro.setItem(key, val);
}
}
}
/* current profile ID pattern - current-profile:xxx */
curProPat = static new RexPattern('current-profile=([^\n]*)\n?$')
/* setting ID pattern for profile items - nnn.xxx=yyy */
proItemPat = static new RexPattern('([^.]+)%.([^=]+)=([^\n]*)\n?$')
/* save the current settings to the user's config file */
saveSettings()
{
/* if there's no profile table, there's nothing to write */
if (profileTab == nil)
return;
try
{
/* open the preferences file; do nothing if that fails */
local f = openSettingsFile(FileAccessWrite);
if (f == nil)
return;
/* write the current profile ID */
f.writeFile('current-profile=<<curProfile>>\n');
/* write the profiles */
profileTab.forEachAssoc(function(id, pro)
{
/* write each element of this profile */
pro.forEach(function(key, val)
{
/* write this profile item */
f.writeFile('<<id>>.<<key>>=<<val>>\n');
});
});
/* done with the file */
f.closeFile();
}
catch (Exception e)
{
/*
* couldn't save the file; this isn't fatal, so just log an
* error event
*/
webMainWin.postSyntheticEvent(
'logError', 'An error occurred saving your setting changes.
(Details: <<e.getExceptionMessage()>>)');
}
}
/* open the settings file */
openSettingsFile(access)
{
/* get the filename; abort if we don't have a file */
local name = getSettingsFile();
if (name == nil)
return nil;
/* open the file */
try
{
/* open the file and return the handle */
return File.openTextFile(name, access, 'ascii');
}
catch (Exception exc)
{
/* failed to open the file */
return nil;
}
}
/* get the settings file path */
getSettingsFile()
{
/* if we're in local stand-alone mode, use the local Web UI file */
if (getLaunchHostAddr() == nil)
return WebUIPrefsFile;
/* if there's a storage server session, the file is on the server */
if (clientSession.storageSID != nil)
return '~<<clientSession.storageSID>>/special/2';
/*
* We're in client/server mode, but there's no storage server.
* In this mode, we don't have any server-side location to store
* files, so we can't save or restore the configuration.
*/
return nil;
}
/* get the current settings as XML, to send to the web UI */
getXML()
{
/* if there's no profile table, there are no settings to return */
if (profileTab == nil)
return '';
/* create a buffer for the results */
local s = new StringBuffer();
/* add the current profile */
s.append('<currentProfile><<curProfile>></currentProfile>');
/* add each profile's contents */
profileTab.forEachAssoc(function(id, pro)
{
/* open this profile section */
s.append('<profile><id><<id>></id>');
/* add each style element */
pro.forEach(function(id, val) {
s.append('<item>'
+ '<id><<id.htmlify()>></id>'
+ '<value><<val.htmlify()>></value>'
+ '</item>');
});
/* close this profile's section */
s.append('</profile>');
});
/* return the result as an XML string */
return toString(s);
}
/* the client session for this preference list */
clientSession = nil
/*
* profile table - this is a LookupTable of WebUIProfile objects
* keyed by profile name
*/
profileTab = nil
/* current active profile selected by the user */
curProfile = nil
;
/* ------------------------------------------------------------------------ */
/*
* Status line window
*/
class WebStatusWin: WebWindow
/* my request path and actual resource path */
vpath = '/statwin.html'
src = 'webuires/statwin.html'
/*
* Set the room and score/turns portions of the status line. This
* sets the left side of the status line to the 'room' text (which
* can contain HTML markups), and the right side to the the
* score/turns values, if present. If the turn counter is omitted
* but the score value is present, we'll just show the score value;
* otherwise we'll format these as "score/turns". If no score value
* is present, we'll leave the right side blank.
*/
setStatus(room, score?, turns?)
{
/* set up the room text in the left portion */
local msg = '<<room>>';
/*
* format the right side: 'score/turns', 'score', or empty,
* depending on what the caller specified
*/
local rt = (score != nil ? toString(score) : '')
+ (turns != nil ? '/' + turns : '');
/* if there's a right side, wrap it with right alignment */
if (rt != '')
msg += '<<rt>>';
/*
* our left/right divisions are floats, which some browsers don't
* count against the container size; so add a clear:all division
* to make sure we count the height properly
*/
msg += '<div style="clear: both;"></div>';
/* set the text */
setStatusText(msg);
}
/*
* Set the text of the status line. This sets the entire status
* window to the given HTML text, without any additional formatting.
*/
setStatusText(msg)
{
/* setting new text, so reset the stream context */
sthCtx.resetState();
/* set the new text */
txt_.deleteChars(1);
txt_.append(msg.specialsToHtml(sthCtx));
/* note that we have new text to send to the UI */
deltas_ = true;
}
/* add text to the status line */
write(msg)
{
/* add the text */
txt_.append(msg.specialsToHtml(sthCtx));
/* note that we have new text to send to the UI */
deltas_ = true;
}
/* clear the window */
clearWindow()
{
setStatusText('');
}
/* flush pending text to the window */
flushWin()
{
/* if we have any deltas since the last flush, send changes */
if (deltas_)
{
/*
* Send a set-text event, along with a resize. The resize
* ensures that the status line adjusts to the current
* content size, assuming the window is using automatic
* content-height sizing.
*/
sendWinEvent(
'<text><<toString(txt_).htmlify()>></text><resize/>');
/* reset the deltas counter */
deltas_ = nil;
}
}
/*
* Refigure the window size. The status line is generally set up to
* be automatically sized to its contents, which requires that we
* tell the UI when it's time to recalculate the layout to reflect
* the current contents after a change.
*/
resize() { sendWinEvent('<resize/>'); }
/* get the current state to send to the browser */
getState(client)
{
return '<text><<toString(txt_).htmlify()>></text>';
}
/* do we have any deltas since the last flush? */
deltas_ = nil
/* the current status message */
txt_ = perInstance(new StringBuffer(512))
;
/* ------------------------------------------------------------------------ */
/*
* The standard "main window" of our user interface. This is the game
* object that represents the default initial HTML page that the player's
* web browser connects to. We build this out of three base classes:
*
* - WebResourceInit, because this is the starting page that the browser
* initially connects to. This class does the initial handshaking to set
* up the session.
*
* - WebResourceResFile, because we store the HTML for the page in a
* resource file. This class does the work of sending the resource
* file's contents to the browser.
*
* - WebLayoutWindow, because the default main page is also a layout
* window, which is basically a container for IFRAME elements where we
* plug in the sub-windows that make up the game's user interface.
*
* Games can customize the front page in any way they like. If you want
* to customize the HTML of the main page, you can substitute a different
* HTML (.html) file, and change the processName() method to return the
* name of that file. If you want to use something other than a layout
* window as the front page, you can simply replace this whole class.
*/
transient webMainWin: WebResourceInit, WebLayoutWindow, WebResourceResFile
/*
* match the webuires directory path as the URL path, but map this to
* main.html as the underlying resource name
*/
vpath = '/'
processName(n) { return 'webuires/main.html'; }
/* the top window is always called "main" */
name = 'main'
pathName = 'main'
/* the window title */
title = 'TADS'
/* set the window title */
setTitle(title)
{
/* remember the title */
self.title = title;
/* update the browser */
sendWinEvent('<setTitle><<title.htmlify()>></setTitle>');
}
/*
* Client session for current command line input. Certain modal
* interactions, such as file dialogs, are directed only to the
* client that initiated the current command.
*/
curCmdClient = nil
/* get the state */
getState(client)
{
/* get the inherited layout state */
local s = inherited(client);
/* add the preference settings */
s += '<prefs><<client.uiPrefs.getXML()>></prefs>';
/* add the window title */
s += '<title><<title.htmlify()>></title>';
/* add the input event, input file, and input dialog states */
s += inputEventState;
s += inputDialogState;
s += menuSysState;
/* if this is the command client, include the file dialog state */
if (client == curCmdClient)
s += fileDialogState;
/* add the downloadable file list */
s += client.allDownloads().mapAll(
{ x: x.isReady
? '<offerDownload><<x.resPath.htmlify()>></offerDownload>'
: '' }
).join();
/* return the result */
return s;
}
/* wait for an input event */
getInputEvent(timeout)
{
/* flush buffered output */
flushWin();
/* send the get-event request to the client */
inputEventState = '<getInputEvent/>';
inputEventResult = nil;
sendWinEvent(inputEventState);
/* process network events until we get an input event */
processNetRequests({: inputEventResult != nil }, timeout);
/* check what happened */
if (inputEventResult)
{
/* we got an input event - return it */
return inputEventResult;
}
else
{
/*
* There's no result, so we timed out. Tell the UI to cancel
* the input event mode.
*/
inputEventState = '';
sendWinEvent('<cancelInputEvent/>');
/* return a timeout event */
return [InEvtTimeout];
}
}
/* receive an input event */
receiveInputEvent(req, query)
{
local typ = query['type'];
local param = query['param'];
switch (typ)
{
case 'key':
inputEventResult = [InEvtKey, param];
break;
case 'href':
inputEventResult = [InEvtHref, param];
break;
default:
inputEventResult = [InEvtEof];
break;
}
}
/* show the file selector dialog */
getInputFile(prompt, dialogType, fileType, flags)
{
/* mappings from FileTypeXxx to storage manager type codes */
local typeMap = [
FileTypeLog -> 'log',
FileTypeData -> 'dat',
FileTypeCmd -> 'cmd',
FileTypeText -> 'txt',
FileTypeBin -> 'bin',
FileTypeT3Image -> 't3',
FileTypeT3Save -> 't3v',
* -> '*'];
/* get the storage server session for the command client */
local sid = curCmdClient.storageSID;
/*
* If we have a session, get the URL to the storage server file
* selection dialog. Note that we don't specify the session in
* the URL, since the storage server separately maintains the
* session with the client via a cookie.
*/
local url = nil;
if (sid != nil)
url = getNetStorageURL(
'fileDialog?filetype=<<typeMap[fileType]>>'
+ '&dlgtype=<<dialogType == InFileOpen ? 'open' : 'save'>>'
+ '&prompt=<<prompt.urlEncode()>>'
+ '&ret=<<webSession.getFullUrl('/webui/inputFile')>>');
/*
* if there's no storage server, try using a client PC upload or
* download
*/
if (url == nil)
return getInputFileFromClient(prompt, dialogType, fileType, flags);
/*
* set up the file dialog state description (store it in a
* property, so that we can send it again as part of a state
* request if the client window is reloaded from scratch)
*/
fileDialogState =
'<fileDialog>'
+ '<prompt><<prompt.htmlify()>></prompt>'
+ '<dialogType><<dialogType == InFileOpen
? 'open' : 'save'>></dialogType>'
+ '<fileType><<typeMap[fileType]>></fileType>'
+ '<url><<url.htmlify()>></url>'
+ '</fileDialog>';
/*
* Presume that the dialog will be canceled. If the user clicks
* the close box on the dialog, the dialog will be dismissed
* without ever sending us a result. This counts as a
* cancellation, so it's the default result.
*/
fileDialogResult = [InFileCancel];
/* send the request to the current command client */
sendWinEventTo(fileDialogState, curCmdClient);
/* process network events until the dialog is dismissed */
processNetRequests(
{: fileDialogState == '' || !curCmdClient.isAlive });
/* return the dialog result */
return fileDialogResult;
}
/*
* Get an input file from the client PC. We'll attempt to upload or
* download a file from/to the client PC, using a local temporary
* file for the actual file operations. This is a special form of
* the input file dialog that we use when we're not connected to a
* storage server.
*/
getInputFileFromClient(prompt, dialogType, fileType, flags)
{
/* use the appropriate procedure for Open vs Save */
if (dialogType == InFileOpen)
{
/*
* Opening an existing file. Show a dialog asking the user
* to select a file to upload. If the user uploads a file,
* we'll use the local temp file created by the upload as the
* result of the input dialog.
*/
fileDialogState =
'<uploadFileDialog>'
+ '<prompt><<prompt.htmlify()>></prompt>'
+ '</uploadFileDialog>';
/* presume the user will cancel the dialog */
fileDialogResult = [InFileCancel];
/* send the request to the UI */
sendWinEventTo(fileDialogState, curCmdClient);
/* process network events until the dialog is dismissed */
processNetRequests(
{: fileDialogState == '' || !curCmdClient.isAlive });
/* return the file dialog result */
return fileDialogResult;
}
else if (dialogType == InFileSave)
{
/*
* Saving to a new file. The caller is asking the user to
* choose a name for a file that the caller *intends* to
* create, but hasn't yet created. HTTP can only do a
* download as a transaction, though - we can offer an
* existing file to the user to download, and the user will
* choose the location for the download as part of that
* transaction. We don't have that file yet because the
* standard protocol is to ask the user for the name first,
* then write the selected file.
*
* So, we have to play a little game. We don't ask the user
* to save anything right now, because there's no file to
* offer for download yet. Instead, we silently generate a
* local temporary file name and return that to our caller.
* Our caller will create the temp file and write its data.
*
* The trick is that we wrap this temp file name in a
* DownloadTempFile object. When the caller finishes
* writing, it will call closeFile() on the temp file. The
* system will pass that along to use by calling closeFile()
* on our DownloadTempFile object. At that point, we'll have
* a completed temporary file, so we'll pop up a download box
* at that point offering the file for download to the user.
* The user can then select a local file on the client side,
* and we'll send the temporary file's contents as the
* download to store in the client file.
*/
return [InFileSuccess,
tempFileDownloadPage.addFile(fileType, curCmdClient)];
}
else
{
/* unknown dialog type */
return [InFileFailure];
}
}
/*
* Offer a file for download to the client. 'file' is a
* DownloadTempFile object previously created by a call to
* inputFile().
*/
offerDownload(file)
{
/*
* send the "offer download" event to the client - the client
* will display an iframe with the file as an "attachment", which
* will trigger the browser to offer it as a download
*/
sendWinEventTo(
'<offerDownload><<file.resPath.htmlify()>></offerDownload>',
curCmdClient);
}
/* receive a file selection from the file selector dialog */
receiveFileSelection(req, query)
{
/* get the filename from the query */
local f = query['file'];
local desc = query['desc'];
local cancel = (query['cancel'] != nil);
local err = query['error'];
/*
* Set the file dialog result list. This uses the same format as
* the native inputFile() routine: the first element is the
* status code (Success, Failure, Cancel); on success, the second
* element is the filename string. We add two bits of
*/
if (err != nil)
fileDialogResult = [InFileFailure, err];
else if (cancel || f == nil || f == '')
fileDialogResult = [InFileCancel];
else
{
/* we got a filename - add the SID prefix if necessary */
if (rexMatch('~[^/]+/', f) == nil)
f = '~<<curCmdClient.storageSID>>/<<f>>';
/* success - return the filename */
fileDialogResult = [InFileSuccess, f];
/* if there's a description string, return that as well */
if (desc != nil)
fileDialogResult += desc;
}
}
/* receive notification that the file dialog has been closed */
inputFileDismissed()
{
/* clear the file dialog state */
fileDialogState = '';
}
/* receive a file upload from the file upload dialog */
receiveFileUpload(req, query)
{
/* get the file from the request */
local fields = req.getFormFields(), file;
if (fields == 'overflow')
{
/* error */
fileDialogResult = [InFileFailure, libMessages.webUploadTooBig];
}
else if (fields != nil && (file = fields['file']) != nil)
{
/* got it - save the contents to a local temporary file */
local tmpfile = new TemporaryFile();
local fpTmp = nil;
try
{
/* open the temp file */
fpTmp = File.openRawFile(tmpfile, FileAccessWrite);
/* make sure the input file is in raw binary mode */
local fpIn = file.file;
fpIn.setFileMode(FileModeRaw);
/* copy the contents */
fpTmp.writeBytes(fpIn);
/* close the temp file */
fpTmp.closeFile();
/*
* Set the dialog result to the temp file. The system
* will automatically delete the underlying file system
* object when the garbage collector deletes the
* TemporaryFile.
*/
fileDialogResult = [InFileSuccess, tmpfile];
}
catch (Exception exc)
{
/* error - the dialog failed */
fileDialogResult = [InFileFailure, exc.getExceptionMessage()];
/* close and delete the temporary file */
if (fpTmp != nil)
fpTmp.closeFile();
if (tmpfile != nil)
tmpfile.deleteFile();
}
}
else
{
/* no file - consider it a cancellation */
fileDialogResult = [InFileCancel];
}
/* the dialog is dismissed */
fileDialogState = '';
}
/* show a generic inputDialog dialog */
getInputDialog(icon, prompt, buttons, defaultButton, cancelButton)
{
/*
* if one of the standard button sets was selected, turn it into
* a list of localized button names
*/
switch (buttons)
{
case InDlgOk:
buttons = [libMessages.dlgButtonOk];
break;
case InDlgOkCancel:
buttons = [libMessages.dlgButtonOk, libMessages.dlgButtonCancel];
break;
case InDlgYesNo:
buttons = [libMessages.dlgButtonYes, libMessages.dlgButtonNo];
break;
case InDlgYesNoCancel:
buttons = [libMessages.dlgButtonYes, libMessages.dlgButtonNo,
libMessages.dlgButtonCancel];
break;
}
/* get a suitable localized title corresponding to the icon tyep */
local title = [InDlgIconNone -> libMessages.dlgTitleNone,
InDlgIconWarning -> libMessages.dlgTitleWarning,
InDlgIconInfo -> libMessages.dlgTitleInfo,
InDlgIconQuestion -> libMessages.dlgTitleQuestion,
InDlgIconError -> libMessages.dlgTitleError][icon];
/* build the dialog xml description */
inputDialogState =
'<inputDialog>'
+ '<icon><<icon>></icon>'
+ '<title><<title>></title>'
+ '<prompt><<prompt.htmlify>></prompt>'
+ '<buttons>'
+ buttons.mapAll({b: '<button><<b.htmlify()>></button>'})
.join('')
+ '</buttons>'
+ '<defaultButton><<defaultButton>></defaultButton>'
+ '<cancelButton><<cancelButton>></cancelButton>'
+ '</inputDialog>';
/* send the request to the client */
inputDialogResult = nil;
sendWinEvent(inputDialogState);
/* process network events until we get an answer */
processNetRequests({: inputDialogResult != nil });
/* return the dialog result */
return inputDialogResult;
}
/* receive a selection from the input dialog */
receiveInputDialog(req, query)
{
/* note the result button */
inputDialogResult = toInteger(query['button']);
/* if we didn't get a valid button index, select button 1 */
if (inputDialogResult == 0)
inputDialogResult = 1;
/* the dialog is no longer open */
inputDialogState = '';
}
/*
* Post a synthetic event. A synthetic event looks like a regular UI
* or network event, but is generated internally instead of being
* delivered from the underlying browser or network subsystems.
*
* 'id' is a string giving the event type. The remaining parameters
* are up to each event type to define.
*/
postSyntheticEvent(id, [params])
{
/* add it to the synthetic event queue */
synthEventQueue.append([id] + params);
}
/* is a synthetic event ready? */
syntheticEventReady() { return synthEventQueue.length() > 0; }
/* pull the next synthetic event from the queue */
getSyntheticEvent() { return synthEventQueue.shift(); }
/*
* file dialog state - this is the XML describing the currently open
* file dialog; if the dialog isn't open, this is an empty string
*/
fileDialogState = ''
/*
* file dialog result - this is a result list using the same format
* as the native inputFile() function
*/
fileDialogResult = nil
/*
* input dialog state - this is the XML describing an input dialog
* while a dialog is running, or an empty string if not
*/
inputDialogState = ''
/* input dialog result - this is the button number the user selected */
inputDialogResult = nil
/* input event state */
inputEventState = ''
/* input event result */
inputEventResult = nil
/* menuSys state - menu system state (maintained by the menu module) */
menuSysState = ''
/*
* Synthetic event queue. This is a vector of synthetic events, set
* up in the [type, params...] format that the system inputEvent()
* function and related functions use. The 'type' code for a
* synthetic evente is a string instead of the numeric identifier
* that the system functions use.
*/
synthEventQueue = static new transient Vector()
;
/* ------------------------------------------------------------------------ */
/*
* Temporary file download page. This page serves temporary files
* created via inputFile() as HTTP downloads to the client.
*/
transient tempFileDownloadPage: WebResource
vpath = static new RexPattern('/clienttmp/')
processRequest(req, query)
{
/*
* look up the file - the key in our table is the ID string after
* the /clienttmp/ path prefix
*/
local id = query[1].substr(12);
local client = ClientSession.find(req);
local desc = client.downloads[id];
/* check for cancellation */
if (query['cancel'] != nil)
{
/*
* acknowledge the request - do this before we cancel the
* file, since removing the file will remove the link that
* fired this request (the order probably isn't a big deal
* one way or the other, and we probably can't really control
* it anyway as the browser might process the ack and the
* event out of order, but just in case)
*/
sendAck(req);
/* if we found the descriptor, cancel it */
if (desc != nil)
client.cancelDownload(desc);
/* done */
return;
}
/* if we didn't find it, return failure */
if (desc == nil)
{
req.sendReply(404);
return;
}
/* open the file */
local fp = nil;
try
{
fp = File.openRawFile(desc.tempFileName, FileAccessRead);
}
catch (Exception exc)
{
/* couldn't send the file - send a 404 */
req.sendReply(404);
return;
}
/* set up the download file headers */
local headers = [
'Content-Disposition: attachment; filename=<<desc.resName>>'
];
try
{
/* send the file's contents */
req.sendReply(fp, desc.mimeType, 200, headers);
}
catch (Exception exc)
{
/* ignore errors */
}
/* done with the file */
fp.closeFile();
/*
* We've at least tried sending the file, so remove it from the
* download list. This was just a temp file, so there's no need
* to keep it around even if we ran into an error sending it; if
* the send failed, the user can just repeat the operation that
* generated the file.
*/
client.cancelDownload(desc);
}
/* add a file to our list of downloadable files */
addFile(fileType, client)
{
/*
* Generate a server-side name template based on the file type.
* The name doesn't matter to us, but browsers will display it to
* the user, and many browsers use the server-side name as the
* default name for the newly downloaded file in the "Save File"
* dialog. Many browsers also use the suffix to determine the
* file type, ignoring any Content-Type headers.
*/
local tpl = [FileTypeLog -> 'Script#.txt',
FileTypeData -> 'File#.dat',
FileTypeCmd -> 'Command#.txt',
FileTypeText -> 'File#.txt',
FileTypeBin -> 'File#.bin',
FileTypeT3Image -> 'Story#.t3',
FileTypeT3Save -> 'Save#.t3v',
* -> 'File#'][fileType];
/* figure the mime type based on the file type */
local mimeType = [FileTypeLog -> 'text/plain',
FileTypeCmd -> 'text/plain',
FileTypeT3Image -> 'application/x-t3vm-image',
FileTypeT3Save -> 'application/x-t3vm-state',
* -> 'application/octet-stream'][fileType];
/* replace the '#' in the template with the next ID value */
tpl = tpl.findReplace('#', toString(nextID++));
/* create a new table entry */
local desc = new DownloadTempFile(tpl, mimeType);
/* add this download to each client */
client.addDownload(desc);
/* return the descriptor */
return desc;
}
/* next available ID */
nextID = 1
;
/*
* Downloadable temporary file descriptor. We create this object when
* the program calls inputFile() to ask for a writable file. This lets
* the caller create and write a temporary file on the server side; when
* the caller is done with the file, we'll offer the file for download to
* the client through the UI.
*/
class DownloadTempFile: object
construct(res, mimeType)
{
tempFileName = new TemporaryFile();
resName = res;
resPath = '/clienttmp/<<res>>';
timeCreated = getTime(GetTimeTicks);
self.mimeType = mimeType;
}
/*
* File spec interface. This allows the DownloadTempFile to be used as
* though it were a filename string.
*
* When the object is passed to one of the File.open methods, or to
* saveGame(), setScriptFile(), etc., the system will call our
* getFilename() method to determine the actual underlying file.
* We'll return our temporary file object.
*
* When the underlying file is closed, the system calls our
* closeFile() method to notify us.
*/
getFilename() { return tempFileName; }
closeFile()
{
/* mark the file as ready for download */
isReady = true;
/* offering the file to the client as an HTTP download */
webMainWin.offerDownload(self);
}
/* TemporaryFile object for the local temp file */
tempFileName = nil
/* root resource name, and full resource path */
resName = nil
resPath = nil
/* MIME type */
mimeType = nil
/* creation timestamp, as a system tick count value */
timeCreated = 0
/* is the file ready for download? */
isReady = nil
/* this is a web temp file */
isWebTempFile = true
;
/* ------------------------------------------------------------------------ */
/*
* Input event page. The client javascript does a GET on this resource
* to send us an input event.
*/
inputEventPage: WebResource
vpath = '/webui/inputEvent'
processRequest(req, query)
{
/* send the event to the main window object */
webMainWin.receiveInputEvent(req, query);
/* acknowledge the request */
sendAck(req);
}
;
/* ------------------------------------------------------------------------ */
/*
* Input dialog event page. The web UI sends a GET to this page when the
* user selects a button in an input dialog.
*/
inputDialogPage: WebResource
vpath = '/webui/inputDialog'
processRequest(req, query)
{
/* send the event to the main window */
webMainWin.receiveInputDialog(req, query);
/* acknowledge the request */
sendAck(req);
}
;
/* ------------------------------------------------------------------------ */
/*
* File dialog event page. This page is used by the IFDB Storage Server
* file dialog to return information to the game UI. The IFDB dialog
* page can't itself perform scripting actions on the enclosing dialog
* frame, since it's being served from a different domain - browsers
* prohibit cross-domain scripting for security reasons. The IFDB dialog
* must therefore navigate back to a page within the game server domain
* in order to return information through scripting. This is that page:
* when the IFDB page is ready to return information, it navigates its
* frame to this page, passing the return values in the request
* parameters. Since this page is served by the game server, within the
* game server domain, the browser allows it to use scripting actions on
* its enclosing frame. We finish the job by dismissing the dialog in
* the UI.
*/
inputFilePage: WebResource
vpath = '/webui/inputFile'
processRequest(req, query)
{
/* set the file selection in the main window */
webMainWin.receiveFileSelection(req, query);
/*
* We have our response, so dismiss the file dialog. Do this by
* sending back a page with script instructions to close the
* containing dialog window.
*/
req.sendReply(
'<html><body>'
+ '<script type="text/javascript">'
+ 'window.parent.dismissDialogById("inputFile");'
+ '</script>'
+ '</body></html>');
}
;
/*
* Cancel the input dialog. This is called from the UI directly to
* cancel the file selection, when the user closes the dialog through the
* enclosing main page UI rather than from within the dialog. This is
* useful if the dialog page fails to load, for example.
*
* Note: the upload file dialog also uses this. The upload dialog is
* basically a variation on the regular input file dialog.
*/
inputFileCancel: WebResource
vpath = '/webui/inputFileDismissed'
processRequest(req, query)
{
/* note that the dialog has been dismissed */
webMainWin.inputFileDismissed();
/* acknowledge the request */
sendAck(req);
}
;
/*
* Receive results from the input file dialog
*/
uploadFilePage: WebResource
vpath = '/webui/uploadFileDialog'
processRequest(req, query)
{
/* send the request to the main window */
webMainWin.receiveFileUpload(req, query);
/* send back a script to dismiss the dialog */
req.sendReply(
'<html><body>'
+ '<script type="text/javascript">'
+ 'window.parent.dismissDialogById("uploadFile");'
+ '</script>'
+ '</body></html>');
}
;
/* ------------------------------------------------------------------------ */
/*
* Receive the client's screen name setting
*/
setScreenNamePage: WebResource
vpath = '/webui/setScreenName'
processRequest(req, query)
{
/* set the name in the client session */
local cli = ClientSession.find(req);
if (cli != nil)
cli.screenName = query['name'];
/* acknowledge the request */
sendAck(req);
}
;
TADS 3 Library Manual
Generated on 5/16/2013 from TADS version 3.1.3