Newer
Older
/* js_bbs.cpp */
/* Synchronet JavaScript "bbs" Object */
/* $Id$ */
/****************************************************************************
* @format.tab-size 4 (Plain Text/Source Code File Header) *
* @format.use-tabs true (see http://www.synchro.net/ptsc_hdr.html) *
* *
* Copyright 2009 Rob Swindell - http://www.synchro.net/copyright.html *
* *
* This program is free software; you can redistribute it and/or *
* modify it under the terms of the GNU General Public License *
* as published by the Free Software Foundation; either version 2 *
* of the License, or (at your option) any later version. *
* See the GNU General Public License for more details: gpl.txt or *
* http://www.fsf.org/copyleft/gpl.html *
* *
* Anonymous FTP access to the most recent released source is available at *
* ftp://vert.synchro.net, ftp://cvs.synchro.net and ftp://ftp.synchro.net *
* *
* Anonymous CVS access to the development source and modification history *
* is available at cvs.synchro.net:/cvsroot/sbbs, example: *
* cvs -d :pserver:anonymous@cvs.synchro.net:/cvsroot/sbbs login *
* (just hit return, no password is necessary) *
* cvs -d :pserver:anonymous@cvs.synchro.net:/cvsroot/sbbs checkout src *
* *
* For Synchronet coding style and modification guidelines, see *
* http://www.synchro.net/source.html *
* *
* You are encouraged to submit any modifications (preferably in Unix diff *
* format) via e-mail to mods@synchro.net *
* *
* Note: If this box doesn't appear square, then you need to fix your tabs. *
****************************************************************************/
#include "sbbs.h"
#include "js_request.h"
#ifdef JAVASCRIPT
/*****************************/
/* BBS Object Properites */
/*****************************/
enum {
BBS_PROP_SYS_STATUS
,BBS_PROP_STARTUP_OPT
,BBS_PROP_ANSWER_TIME
,BBS_PROP_LOGON_TIME
,BBS_PROP_START_TIME
,BBS_PROP_NS_TIME
,BBS_PROP_LAST_NS_TIME
,BBS_PROP_ONLINE
,BBS_PROP_TIMELEFT
,BBS_PROP_EVENT_TIME
,BBS_PROP_EVENT_CODE
,BBS_PROP_NODE_NUM
,BBS_PROP_NODE_MISC
,BBS_PROP_NODE_VAL_USER
,BBS_PROP_LOGON_ULB
,BBS_PROP_LOGON_DLB
,BBS_PROP_LOGON_ULS
,BBS_PROP_LOGON_DLS
,BBS_PROP_LOGON_POSTS
,BBS_PROP_LOGON_EMAILS
,BBS_PROP_LOGON_FBACKS
,BBS_PROP_POSTS_READ
,BBS_PROP_MENU_DIR
,BBS_PROP_MENU_FILE
,BBS_PROP_MAIN_CMDS
,BBS_PROP_CURGRP
,BBS_PROP_CURSUB
,BBS_PROP_CURSUB_CODE
,BBS_PROP_CURLIB
,BBS_PROP_CURDIR
,BBS_PROP_CURDIR_CODE
,BBS_PROP_CONNECTION /* READ ONLY */
,BBS_PROP_RLOGIN_NAME
,BBS_PROP_RLOGIN_PASS
,BBS_PROP_ALTUL
,BBS_PROP_ERRORLEVEL /* READ ONLY */
/* READ ONLY */
,BBS_PROP_SMB_GROUP
,BBS_PROP_SMB_GROUP_DESC
,BBS_PROP_SMB_GROUP_NUM
,BBS_PROP_SMB_SUB
,BBS_PROP_SMB_SUB_DESC
,BBS_PROP_SMB_SUB_CODE
,BBS_PROP_SMB_SUB_NUM
,BBS_PROP_SMB_ATTR
,BBS_PROP_SMB_LAST_MSG
,BBS_PROP_SMB_TOTAL_MSGS
,BBS_PROP_SMB_MSGS
,BBS_PROP_SMB_CURMSG
/* READ ONLY */
,BBS_PROP_MSG_TO
,BBS_PROP_MSG_TO_EXT
,BBS_PROP_MSG_TO_NET
,BBS_PROP_MSG_TO_AGENT
,BBS_PROP_MSG_FROM
,BBS_PROP_MSG_FROM_EXT
,BBS_PROP_MSG_FROM_NET
,BBS_PROP_MSG_FROM_AGENT
,BBS_PROP_MSG_REPLYTO
,BBS_PROP_MSG_REPLYTO_EXT
,BBS_PROP_MSG_REPLYTO_NET
,BBS_PROP_MSG_REPLYTO_AGENT
,BBS_PROP_MSG_SUBJECT
,BBS_PROP_MSG_DATE
,BBS_PROP_MSG_TIMEZONE
,BBS_PROP_MSG_DATE_IMPORTED
,BBS_PROP_MSG_ATTR
,BBS_PROP_MSG_AUXATTR
,BBS_PROP_MSG_NETATTR
,BBS_PROP_MSG_OFFSET
,BBS_PROP_MSG_NUMBER
,BBS_PROP_MSG_EXPIRATION
,BBS_PROP_MSG_FORWARDED
,BBS_PROP_MSG_THREAD_BACK
,BBS_PROP_MSG_THREAD_NEXT
,BBS_PROP_MSG_THREAD_FIRST
,BBS_PROP_MSG_ID
,BBS_PROP_MSG_REPLY_ID
/* READ ONLY */
,BBS_PROP_BATCH_UPLOAD_TOTAL
,BBS_PROP_BATCH_DNLOAD_TOTAL
,BBS_PROP_COMMAND_STR
#ifdef BUILD_JSDOCS
"system status bitfield (see <tt>SS_*</tt> in <tt>sbbsdefs.js</tt> for bit definitions)"
,"startup options bitfield (see <tt>BBS_OPT_*</tt> in <tt>sbbsdefs.js</tt> for bit definitions)"
,"answer time, in <i>time_t</i> format"
,"logon time, in <i>time_t</i> format"
,"time from which user's time left is calculated, in <i>time_t</i> format"
,"current file new-scan time, in <i>time_t</i> format"
,"previous file new-scan time, in <i>time_t</i> format"
,"online (see <tt>ON_*</tt> in <tt>sbbsdefs.js</tt> for valid values)"
,"time left (in seconds)"
,"time of next exclusive event (in <i>time_t</i> format), or 0 if none"
,"internal code of next exclusive event"
,"current node settings bitfield (see <tt>NM_*</tt> in <tt>sbbsdefs.js</tt> for bit definitions)"
,"current node action (see <tt>nodedefs.js</tt> for valid values)"
,"validation feedback user for this node (or 0 for no validation feedback required)"
,"bytes uploaded during this session"
,"bytes downloaded during this session"
,"files uploaded during this session"
,"files downloaded during this session"
,"messages posted during this session"
,"e-mails sent during this session"
,"feedback messages sent during this session"
,"messages read during this session"
,"menu subdirectory (overrides default)"
,"menu file (overrides default)"
,"total main menu commands received from user during this session"
,"total file menu commands received from user during this session"
,"current message group"
,"current message sub-board"
,"current message sub-board internal code"
,"current file library"
,"current file directory"
,"current file directory internal code"
,"login name given during RLogin negotiation"
,"password specified during RLogin negotiation"
,"client name"
,"current alternate upload path number"
,"error level returned from last executed external program"
,"message group name of message being read"
,"message group description of message being read"
,"message group number of message being read"
,"sub-board name of message being read"
,"sub-board description of message being read"
,"sub-board internal code of message being read"
,"sub-board number of message being read"
,"message base attributes"
,"highest message number in message base"
,"total number of messages in message base"
,"number of messages loaded from message base"
,"current message number in message base"
,"message recipient name"
,"message recipient extension"
,"message recipient network type"
,"message recipient agent type"
,"message sender name"
,"message sender extension"
,"message sender network type"
,"message sender agent type"
,"message reply-to name"
,"message reply-to extension"
,"message reply-to network type"
,"message reply-to agent type"
,"message subject"
,"message date/time"
,"message time zone"
,"message date/time imported"
,"message attributes"
,"message auxillary attributes"
,"message network attributes"
,"message header offset"
,"message number"
,"message expiration"
,"message forwarded"
,"message thread, back message number (AKA msg_thread_orig)"
,"message thread, next message number"
,"message thread, first reply to this message"
,"message identifier"
,"message replied-to identifier"
,"message delivery attempt counter"
,"number of files in batch upload queue"
,"number of files in batch download queue"
,"current command shell/module <i>command string</i> value"
static JSBool js_bbs_get(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
{
char tmp[128];
const char* p=NULL;
const char* nulstr="";
if((sbbs=(sbbs_t*)JS_GetContextPrivate(cx))==NULL)
return(JS_FALSE);
tiny = JSVAL_TO_INT(id);
switch(tiny) {
case BBS_PROP_SYS_STATUS:
val=sbbs->sys_status;
break;
case BBS_PROP_STARTUP_OPT:
val=sbbs->startup->options;
break;
case BBS_PROP_ANSWER_TIME:
val=sbbs->answertime;
break;
case BBS_PROP_LOGON_TIME:
val=sbbs->logontime;
break;
case BBS_PROP_START_TIME:
val=sbbs->starttime;
break;
case BBS_PROP_NS_TIME:
val=sbbs->ns_time;
break;
case BBS_PROP_LAST_NS_TIME:
val=sbbs->last_ns_time;
break;
case BBS_PROP_ONLINE:
val=sbbs->online;
break;
rc=JS_SUSPENDREQUEST(cx);
val=sbbs->gettimeleft(false);
JS_RESUMEREQUEST(cx, rc);
case BBS_PROP_EVENT_TIME:
val=sbbs->event_time;
break;
case BBS_PROP_EVENT_CODE:
p=sbbs->event_code;
break;
case BBS_PROP_NODE_NUM:
val=sbbs->cfg.node_num;
case BBS_PROP_NODE_MISC:
val=sbbs->cfg.node_misc;
case BBS_PROP_NODE_ACTION:
val=sbbs->action;
break;
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
case BBS_PROP_NODE_VAL_USER:
val=sbbs->cfg.node_valuser;
break;
case BBS_PROP_LOGON_ULB:
val=sbbs->logon_ulb;
break;
case BBS_PROP_LOGON_DLB:
val=sbbs->logon_dlb;
break;
case BBS_PROP_LOGON_ULS:
val=sbbs->logon_uls;
break;
case BBS_PROP_LOGON_DLS:
val=sbbs->logon_dls;
break;
case BBS_PROP_LOGON_POSTS:
val=sbbs->logon_posts;
break;
case BBS_PROP_LOGON_EMAILS:
val=sbbs->logon_emails;
break;
case BBS_PROP_LOGON_FBACKS:
val=sbbs->logon_fbacks;
break;
case BBS_PROP_POSTS_READ:
val=sbbs->posts_read;
break;
case BBS_PROP_MENU_DIR:
p=sbbs->menu_dir;
case BBS_PROP_MENU_FILE:
p=sbbs->menu_file;
case BBS_PROP_MAIN_CMDS:
val=sbbs->main_cmds;
case BBS_PROP_CURGRP:
val=sbbs->curgrp;
break;
case BBS_PROP_CURSUB:
if(sbbs->curgrp<sbbs->usrgrps)
val=sbbs->cursub[sbbs->curgrp];
break;
case BBS_PROP_CURSUB_CODE:
if(sbbs->cursubnum<sbbs->cfg.total_subs)
p=sbbs->cfg.sub[sbbs->cursubnum]->code;
break;
case BBS_PROP_CURLIB:
val=sbbs->curlib;
break;
case BBS_PROP_CURDIR:
if(sbbs->curlib<sbbs->usrlibs)
val=sbbs->curdir[sbbs->curlib];
break;
case BBS_PROP_CURDIR_CODE:
if(sbbs->curdirnum<sbbs->cfg.total_dirs)
p=sbbs->cfg.dir[sbbs->curdirnum]->code;
break;
case BBS_PROP_CONNECTION:
p=sbbs->connection;
case BBS_PROP_RLOGIN_NAME:
p=sbbs->rlogin_name;
break;
case BBS_PROP_RLOGIN_PASS:
p=sbbs->rlogin_pass;
break;
case BBS_PROP_CLIENT_NAME:
p=sbbs->client_name;
break;
case BBS_PROP_ALTUL:
val=sbbs->altul;
break;
case BBS_PROP_ERRORLEVEL:
val=sbbs->errorlevel;
break;
/* Currently Open Message Base (sbbs.smb) */
case BBS_PROP_SMB_GROUP:
if(sbbs->smb.subnum==INVALID_SUB || sbbs->smb.subnum>=sbbs->cfg.total_subs)
p=nulstr;
else
p=sbbs->cfg.grp[sbbs->cfg.sub[sbbs->smb.subnum]->grp]->sname;
break;
case BBS_PROP_SMB_GROUP_DESC:
if(sbbs->smb.subnum==INVALID_SUB || sbbs->smb.subnum>=sbbs->cfg.total_subs)
p=nulstr;
else
p=sbbs->cfg.grp[sbbs->cfg.sub[sbbs->smb.subnum]->grp]->lname;
break;
case BBS_PROP_SMB_GROUP_NUM:
if(sbbs->smb.subnum!=INVALID_SUB && sbbs->smb.subnum<sbbs->cfg.total_subs) {
uint ugrp;
for(ugrp=0;ugrp<sbbs->usrgrps;ugrp++)
if(sbbs->usrgrp[ugrp]==sbbs->cfg.sub[sbbs->smb.subnum]->grp)
break;
val=ugrp+1;
}
break;
case BBS_PROP_SMB_SUB:
if(sbbs->smb.subnum==INVALID_SUB || sbbs->smb.subnum>=sbbs->cfg.total_subs)
p=nulstr;
else
p=sbbs->cfg.sub[sbbs->smb.subnum]->sname;
break;
case BBS_PROP_SMB_SUB_DESC:
if(sbbs->smb.subnum==INVALID_SUB || sbbs->smb.subnum>=sbbs->cfg.total_subs)
p=nulstr;
else
p=sbbs->cfg.sub[sbbs->smb.subnum]->lname;
break;
case BBS_PROP_SMB_SUB_CODE:
if(sbbs->smb.subnum==INVALID_SUB || sbbs->smb.subnum>=sbbs->cfg.total_subs)
p=nulstr;
else
p=sbbs->cfg.sub[sbbs->smb.subnum]->code;
break;
case BBS_PROP_SMB_SUB_NUM:
if(sbbs->smb.subnum!=INVALID_SUB && sbbs->smb.subnum<sbbs->cfg.total_subs) {
uint ugrp;
for(ugrp=0;ugrp<sbbs->usrgrps;ugrp++)
if(sbbs->usrgrp[ugrp]==sbbs->cfg.sub[sbbs->smb.subnum]->grp)
break;
uint usub;
for(usub=0;usub<sbbs->usrsubs[ugrp];usub++)
if(sbbs->usrsub[ugrp][usub]==sbbs->smb.subnum)
break;
val=usub+1;
}
break;
case BBS_PROP_SMB_ATTR:
val=sbbs->smb.status.attr;
break;
case BBS_PROP_SMB_LAST_MSG:
val=sbbs->smb.status.last_msg;
break;
case BBS_PROP_SMB_TOTAL_MSGS:
val=sbbs->smb.status.total_msgs;
break;
case BBS_PROP_SMB_MSGS:
val=sbbs->smb.msgs;
break;
case BBS_PROP_SMB_CURMSG:
val=sbbs->smb.curmsg;
break;
/* Currently Displayed Message Header (sbbs.current_msg) */
case BBS_PROP_MSG_TO:
if(sbbs->current_msg==NULL || sbbs->current_msg->to==NULL)
p=nulstr;
else
p=sbbs->current_msg->to;
break;
case BBS_PROP_MSG_TO_EXT:
if(sbbs->current_msg==NULL || sbbs->current_msg->to_ext==NULL)
p=nulstr;
else
p=sbbs->current_msg->to_ext;
break;
case BBS_PROP_MSG_TO_NET:
if(sbbs->current_msg==NULL || sbbs->current_msg->to_net.type==NET_NONE)
p=nulstr;
else
p=smb_netaddrstr(&sbbs->current_msg->to_net,tmp);
break;
case BBS_PROP_MSG_TO_AGENT:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->to_agent;
break;
case BBS_PROP_MSG_FROM:
if(sbbs->current_msg==NULL || sbbs->current_msg->from==NULL)
p=nulstr;
else
p=sbbs->current_msg->from;
break;
case BBS_PROP_MSG_FROM_EXT:
if(sbbs->current_msg==NULL || sbbs->current_msg->from_ext==NULL)
p=nulstr;
else
p=sbbs->current_msg->from_ext;
break;
case BBS_PROP_MSG_FROM_NET:
if(sbbs->current_msg==NULL || sbbs->current_msg->from_net.type==NET_NONE)
p=nulstr;
else
p=smb_netaddrstr(&sbbs->current_msg->from_net,tmp);
break;
case BBS_PROP_MSG_FROM_AGENT:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->from_agent;
break;
case BBS_PROP_MSG_REPLYTO:
if(sbbs->current_msg==NULL || sbbs->current_msg->replyto==NULL)
p=nulstr;
else
p=sbbs->current_msg->replyto;
break;
case BBS_PROP_MSG_REPLYTO_EXT:
if(sbbs->current_msg==NULL || sbbs->current_msg->replyto_ext==NULL)
p=nulstr;
else
p=sbbs->current_msg->replyto_ext;
break;
case BBS_PROP_MSG_REPLYTO_NET:
if(sbbs->current_msg==NULL || sbbs->current_msg->replyto_net.type==NET_NONE)
p=nulstr;
else
p=smb_netaddrstr(&sbbs->current_msg->replyto_net,tmp);
break;
case BBS_PROP_MSG_REPLYTO_AGENT:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->replyto_agent;
break;
case BBS_PROP_MSG_SUBJECT:
if(sbbs->current_msg==NULL || sbbs->current_msg->subj==NULL)
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
p=nulstr;
else
p=sbbs->current_msg->subj;
break;
case BBS_PROP_MSG_DATE:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->hdr.when_written.time;
break;
case BBS_PROP_MSG_TIMEZONE:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->hdr.when_written.zone;
break;
case BBS_PROP_MSG_DATE_IMPORTED:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->hdr.when_imported.time;
break;
case BBS_PROP_MSG_ATTR:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->hdr.attr;
break;
case BBS_PROP_MSG_AUXATTR:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->hdr.auxattr;
break;
case BBS_PROP_MSG_NETATTR:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->hdr.netattr;
break;
case BBS_PROP_MSG_OFFSET:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->offset;
break;
case BBS_PROP_MSG_NUMBER:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->hdr.number;
break;
case BBS_PROP_MSG_EXPIRATION:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->expiration;
break;
case BBS_PROP_MSG_FORWARDED:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->forwarded;
break;
case BBS_PROP_MSG_THREAD_BACK:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->hdr.thread_back;
break;
case BBS_PROP_MSG_THREAD_NEXT:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->hdr.thread_next;
break;
case BBS_PROP_MSG_THREAD_FIRST:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->hdr.thread_first;
break;
case BBS_PROP_MSG_DELIVERY_ATTEMPTS:
if(sbbs->current_msg!=NULL)
val=sbbs->current_msg->hdr.delivery_attempts;
break;
case BBS_PROP_MSG_ID:
if(sbbs->current_msg==NULL || sbbs->current_msg->id==NULL)
p=nulstr;
else
p=sbbs->current_msg->id;
break;
case BBS_PROP_MSG_REPLY_ID:
if(sbbs->current_msg==NULL || sbbs->current_msg->reply_id==NULL)
p=nulstr;
else
p=sbbs->current_msg->reply_id;
break;
case BBS_PROP_BATCH_UPLOAD_TOTAL:
val=sbbs->batup_total;
break;
case BBS_PROP_BATCH_DNLOAD_TOTAL:
val=sbbs->batdn_total;
case BBS_PROP_COMMAND_STR:
p=sbbs->main_csi.str;
break;
if(p!=NULL) {
JSString* js_str=JS_NewStringCopyZ(cx, p);
if(js_str==NULL)
return(JS_FALSE);
*vp = STRING_TO_JSVAL(js_str);
} else
return(JS_TRUE);
}
static JSBool js_bbs_set(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
{
sbbs_t* sbbs;
if((sbbs=(sbbs_t*)JS_GetContextPrivate(cx))==NULL)
return(JS_FALSE);
tiny = JSVAL_TO_INT(id);
if(JSVAL_IS_NUM(*vp) || JSVAL_IS_BOOLEAN(*vp))
else if(JSVAL_IS_STRING(*vp)) {
if((js_str = JS_ValueToString(cx, *vp))==NULL)
return(JS_FALSE);
p=JS_GetStringBytes(js_str);
}
case BBS_PROP_SYS_STATUS:
sbbs->sys_status=val;
break;
case BBS_PROP_STARTUP_OPT:
sbbs->startup->options=val;
break;
case BBS_PROP_ANSWER_TIME:
sbbs->answertime=val;
break;
case BBS_PROP_LOGON_TIME:
sbbs->logontime=val;
break;
case BBS_PROP_START_TIME:
sbbs->starttime=val;
break;
case BBS_PROP_NS_TIME:
sbbs->ns_time=val;
break;
case BBS_PROP_LAST_NS_TIME:
sbbs->last_ns_time=val;
break;
case BBS_PROP_ONLINE:
sbbs->online=val;
break;
case BBS_PROP_NODE_MISC:
sbbs->cfg.node_misc=val;
break;
case BBS_PROP_NODE_ACTION:
sbbs->action=(uchar)val;
break;
case BBS_PROP_NODE_VAL_USER:
sbbs->cfg.node_valuser=(ushort)val;
break;
case BBS_PROP_LOGON_ULB:
sbbs->logon_ulb=val;
break;
case BBS_PROP_LOGON_DLB:
sbbs->logon_dlb=val;
break;
case BBS_PROP_LOGON_ULS:
sbbs->logon_uls=val;
break;
case BBS_PROP_LOGON_DLS:
sbbs->logon_dls=val;
break;
case BBS_PROP_LOGON_POSTS:
sbbs->logon_posts=val;
break;
case BBS_PROP_LOGON_EMAILS:
sbbs->logon_emails=val;
case BBS_PROP_LOGON_FBACKS:
sbbs->logon_fbacks=val;
case BBS_PROP_POSTS_READ:
sbbs->posts_read=val;
SAFECOPY(sbbs->menu_dir,p);
SAFECOPY(sbbs->menu_file,p);
case BBS_PROP_MAIN_CMDS:
sbbs->main_cmds=val;
sbbs->xfer_cmds=val;
break;
case BBS_PROP_CURGRP:
if(p!=NULL) { /* set by name */
uint i;
for(i=0;i<sbbs->usrgrps;i++)
if(!stricmp(sbbs->cfg.grp[sbbs->usrgrp[i]]->sname,p))
break;
if(i<sbbs->usrgrps)
sbbs->curgrp=i;
break;
}
if((uint)val<sbbs->cfg.total_grps && (uint)val<sbbs->usrgrps)
sbbs->curgrp=val;
break;
case BBS_PROP_CURSUB:
case BBS_PROP_CURSUB_CODE:
if(p!=NULL) { /* set by code */
for(uint i=0;i<sbbs->usrgrps;i++)
for(uint j=0;j<sbbs->usrsubs[i];j++)
if(!stricmp(sbbs->cfg.sub[sbbs->usrsub[i][j]]->code,p)) {
sbbs->curgrp=i;
sbbs->cursub[i]=j;
break;
}
break;
}
if(sbbs->curgrp<sbbs->cfg.total_grps && (uint)val<sbbs->usrsubs[sbbs->curgrp])
sbbs->cursub[sbbs->curgrp]=val;
break;
case BBS_PROP_CURLIB:
if(p!=NULL) { /* set by name */
uint i;
for(i=0;i<sbbs->usrlibs;i++)
if(!stricmp(sbbs->cfg.lib[sbbs->usrlib[i]]->sname,p))
break;
if(i<sbbs->usrlibs)
sbbs->curlib=i;
break;
}
if((uint)val<sbbs->cfg.total_libs && (uint)val<sbbs->usrlibs)
sbbs->curlib=val;
break;
case BBS_PROP_CURDIR:
case BBS_PROP_CURDIR_CODE:
if(p!=NULL) { /* set by code */
for(uint i=0;i<sbbs->usrlibs;i++)
for(uint j=0;j<sbbs->usrdirs[i];j++)
if(!stricmp(sbbs->cfg.dir[sbbs->usrdir[i][j]]->code,p)) {
sbbs->curlib=i;
sbbs->curdir[i]=j;
break;
}
break;
}
if(sbbs->curlib<sbbs->cfg.total_libs && (uint)val<sbbs->usrdirs[sbbs->curlib])
sbbs->curdir[sbbs->curlib]=val;
break;
SAFECOPY(sbbs->rlogin_name,p);
case BBS_PROP_RLOGIN_PASS:
SAFECOPY(sbbs->rlogin_pass,p);
break;
SAFECOPY(sbbs->client_name,p);
case BBS_PROP_ALTUL:
if(val<sbbs->cfg.altpaths)
sbbs->altul=(ushort)val;
break;
case BBS_PROP_COMMAND_STR:
sprintf(sbbs->main_csi.str, "%.*s", 1024, p);
break;
default:
return(JS_TRUE);
}
if(sbbs->usrgrps)
sbbs->cursubnum=sbbs->usrsub[sbbs->curgrp][sbbs->cursub[sbbs->curgrp]]; /* Used for ARS */
else
sbbs->cursubnum=INVALID_SUB;
if(sbbs->usrlibs)
sbbs->curdirnum=sbbs->usrdir[sbbs->curlib][sbbs->curdir[sbbs->curlib]]; /* Used for ARS */
else
sbbs->curdirnum=INVALID_DIR;
#define PROP_READONLY JSPROP_ENUMERATE|JSPROP_READONLY
static jsSyncPropertySpec js_bbs_properties[] = {
/* name ,tinyid ,flags ,ver */
{ "sys_status" ,BBS_PROP_SYS_STATUS ,JSPROP_ENUMERATE ,310},
{ "startup_options" ,BBS_PROP_STARTUP_OPT ,JSPROP_ENUMERATE ,310},
{ "answer_time" ,BBS_PROP_ANSWER_TIME ,JSPROP_ENUMERATE ,310},
{ "logon_time" ,BBS_PROP_LOGON_TIME ,JSPROP_ENUMERATE ,310},
{ "start_time" ,BBS_PROP_START_TIME ,JSPROP_ENUMERATE ,314},
{ "new_file_time" ,BBS_PROP_NS_TIME ,JSPROP_ENUMERATE ,310},
{ "last_new_file_time",BBS_PROP_LAST_NS_TIME ,JSPROP_ENUMERATE ,310},
{ "online" ,BBS_PROP_ONLINE ,JSPROP_ENUMERATE ,310},
{ "timeleft" ,BBS_PROP_TIMELEFT ,JSPROP_READONLY ,310}, /* alias */
{ "time_left" ,BBS_PROP_TIMELEFT ,PROP_READONLY ,311},
{ "event_time" ,BBS_PROP_EVENT_TIME ,PROP_READONLY ,311},
{ "event_code" ,BBS_PROP_EVENT_CODE ,PROP_READONLY ,311},
{ "node_num" ,BBS_PROP_NODE_NUM ,PROP_READONLY ,310},
{ "node_settings" ,BBS_PROP_NODE_MISC ,JSPROP_ENUMERATE ,310},
{ "node_action" ,BBS_PROP_NODE_ACTION ,JSPROP_ENUMERATE ,310},
{ "node_val_user" ,BBS_PROP_NODE_VAL_USER ,JSPROP_ENUMERATE ,310},
{ "logon_ulb" ,BBS_PROP_LOGON_ULB ,JSPROP_ENUMERATE ,310},
{ "logon_dlb" ,BBS_PROP_LOGON_DLB ,JSPROP_ENUMERATE ,310},
{ "logon_uls" ,BBS_PROP_LOGON_ULS ,JSPROP_ENUMERATE ,310},
{ "logon_dls" ,BBS_PROP_LOGON_DLS ,JSPROP_ENUMERATE ,310},
{ "logon_posts" ,BBS_PROP_LOGON_POSTS ,JSPROP_ENUMERATE ,310},
{ "logon_emails" ,BBS_PROP_LOGON_EMAILS ,JSPROP_ENUMERATE ,310},
{ "logon_fbacks" ,BBS_PROP_LOGON_FBACKS ,JSPROP_ENUMERATE ,310},
{ "posts_read" ,BBS_PROP_POSTS_READ ,JSPROP_ENUMERATE ,310},
{ "menu_dir" ,BBS_PROP_MENU_DIR ,JSPROP_ENUMERATE ,310},
{ "menu_file" ,BBS_PROP_MENU_FILE ,JSPROP_ENUMERATE ,310},
{ "main_cmds" ,BBS_PROP_MAIN_CMDS ,JSPROP_ENUMERATE ,310},
{ "file_cmds" ,BBS_PROP_FILE_CMDS ,JSPROP_ENUMERATE ,310},
{ "curgrp" ,BBS_PROP_CURGRP ,JSPROP_ENUMERATE ,310},
{ "cursub" ,BBS_PROP_CURSUB ,JSPROP_ENUMERATE ,310},
{ "cursub_code" ,BBS_PROP_CURSUB_CODE ,JSPROP_ENUMERATE ,314},
{ "curlib" ,BBS_PROP_CURLIB ,JSPROP_ENUMERATE ,310},
{ "curdir" ,BBS_PROP_CURDIR ,JSPROP_ENUMERATE ,310},
{ "curdir_code" ,BBS_PROP_CURDIR_CODE ,JSPROP_ENUMERATE ,314},
{ "connection" ,BBS_PROP_CONNECTION ,PROP_READONLY ,310},
{ "rlogin_name" ,BBS_PROP_RLOGIN_NAME ,JSPROP_ENUMERATE ,310},
{ "rlogin_password" ,BBS_PROP_RLOGIN_PASS ,JSPROP_ENUMERATE ,315},
{ "client_name" ,BBS_PROP_CLIENT_NAME ,JSPROP_ENUMERATE ,310},
{ "alt_ul_dir" ,BBS_PROP_ALTUL ,JSPROP_ENUMERATE ,310},
{ "errorlevel" ,BBS_PROP_ERRORLEVEL ,PROP_READONLY ,312},
{ "smb_group" ,BBS_PROP_SMB_GROUP ,PROP_READONLY ,310},
{ "smb_group_desc" ,BBS_PROP_SMB_GROUP_DESC ,PROP_READONLY ,310},
{ "smb_group_number" ,BBS_PROP_SMB_GROUP_NUM ,PROP_READONLY ,310},
{ "smb_sub" ,BBS_PROP_SMB_SUB ,PROP_READONLY ,310},
{ "smb_sub_desc" ,BBS_PROP_SMB_SUB_DESC ,PROP_READONLY ,310},
{ "smb_sub_code" ,BBS_PROP_SMB_SUB_CODE ,PROP_READONLY ,310},
{ "smb_sub_number" ,BBS_PROP_SMB_SUB_NUM ,PROP_READONLY ,310},
{ "smb_attr" ,BBS_PROP_SMB_ATTR ,PROP_READONLY ,310},
{ "smb_last_msg" ,BBS_PROP_SMB_LAST_MSG ,PROP_READONLY ,310},
{ "smb_total_msgs" ,BBS_PROP_SMB_TOTAL_MSGS ,PROP_READONLY ,310},
{ "smb_msgs" ,BBS_PROP_SMB_MSGS ,PROP_READONLY ,310},
{ "smb_curmsg" ,BBS_PROP_SMB_CURMSG ,PROP_READONLY ,310},
{ "msg_to" ,BBS_PROP_MSG_TO ,PROP_READONLY ,310},
{ "msg_to_ext" ,BBS_PROP_MSG_TO_EXT ,PROP_READONLY ,310},
{ "msg_to_net" ,BBS_PROP_MSG_TO_NET ,PROP_READONLY ,310},
{ "msg_to_agent" ,BBS_PROP_MSG_TO_AGENT ,PROP_READONLY ,310},
{ "msg_from" ,BBS_PROP_MSG_FROM ,PROP_READONLY ,310},
{ "msg_from_ext" ,BBS_PROP_MSG_FROM_EXT ,PROP_READONLY ,310},
{ "msg_from_net" ,BBS_PROP_MSG_FROM_NET ,PROP_READONLY ,310},
{ "msg_from_agent" ,BBS_PROP_MSG_FROM_AGENT ,PROP_READONLY ,310},
{ "msg_replyto" ,BBS_PROP_MSG_REPLYTO ,PROP_READONLY ,310},
{ "msg_replyto_ext" ,BBS_PROP_MSG_REPLYTO_EXT ,PROP_READONLY ,310},
{ "msg_replyto_net" ,BBS_PROP_MSG_REPLYTO_NET ,PROP_READONLY ,310},
{ "msg_replyto_agent" ,BBS_PROP_MSG_REPLYTO_AGENT ,PROP_READONLY ,310},
{ "msg_subject" ,BBS_PROP_MSG_SUBJECT ,PROP_READONLY ,310},
{ "msg_date" ,BBS_PROP_MSG_DATE ,PROP_READONLY ,310},
{ "msg_timezone" ,BBS_PROP_MSG_TIMEZONE ,PROP_READONLY ,310},
{ "msg_date_imported" ,BBS_PROP_MSG_DATE_IMPORTED ,PROP_READONLY ,310},
{ "msg_attr" ,BBS_PROP_MSG_ATTR ,PROP_READONLY ,310},
{ "msg_auxattr" ,BBS_PROP_MSG_AUXATTR ,PROP_READONLY ,310},
{ "msg_netattr" ,BBS_PROP_MSG_NETATTR ,PROP_READONLY ,310},
{ "msg_offset" ,BBS_PROP_MSG_OFFSET ,PROP_READONLY ,310},
{ "msg_number" ,BBS_PROP_MSG_NUMBER ,PROP_READONLY ,310},
{ "msg_expiration" ,BBS_PROP_MSG_EXPIRATION ,PROP_READONLY ,310},
{ "msg_forwarded" ,BBS_PROP_MSG_FORWARDED ,PROP_READONLY ,310},
{ "msg_thread_back" ,BBS_PROP_MSG_THREAD_BACK ,PROP_READONLY ,312},
{ "msg_thread_orig" ,BBS_PROP_MSG_THREAD_BACK ,JSPROP_READONLY,310}, /* alias */
{ "msg_thread_next" ,BBS_PROP_MSG_THREAD_NEXT ,PROP_READONLY ,310},
{ "msg_thread_first" ,BBS_PROP_MSG_THREAD_FIRST ,PROP_READONLY ,310},
{ "msg_id" ,BBS_PROP_MSG_ID ,PROP_READONLY ,310},
{ "msg_reply_id" ,BBS_PROP_MSG_REPLY_ID ,PROP_READONLY ,310},
{ "msg_delivery_attempts" ,BBS_PROP_MSG_DELIVERY_ATTEMPTS
,PROP_READONLY ,310},
{ "batch_upload_total",BBS_PROP_BATCH_UPLOAD_TOTAL,PROP_READONLY ,310},
{ "batch_dnload_total",BBS_PROP_BATCH_DNLOAD_TOTAL,PROP_READONLY ,310},
/* Utility functions */
static uint get_subnum(JSContext* cx, sbbs_t* sbbs, jsval val)
{
uint subnum=INVALID_SUB;
if(JSVAL_IS_STRING(val)) {
char* p=JS_GetStringBytes(JS_ValueToString(cx,val));
for(subnum=0;subnum<sbbs->cfg.total_subs;subnum++)
if(!stricmp(sbbs->cfg.sub[subnum]->code,p))
break;
} else if(JSVAL_IS_NUM(val))
JS_ValueToInt32(cx,val,(int32*)&subnum);
else if(sbbs->usrgrps>0)
subnum=sbbs->usrsub[sbbs->curgrp][sbbs->cursub[sbbs->curgrp]];
return(subnum);
}
static uint get_dirnum(JSContext* cx, sbbs_t* sbbs, jsval val)
{
uint dirnum=INVALID_DIR;
if(JSVAL_IS_STRING(val)) {
char* p=JS_GetStringBytes(JS_ValueToString(cx,val));
for(dirnum=0;dirnum<sbbs->cfg.total_dirs;dirnum++)
if(!stricmp(sbbs->cfg.dir[dirnum]->code,p))
break;
} else if(JSVAL_IS_NUM(val))
JS_ValueToInt32(cx,val,(int32*)&dirnum);
else if(sbbs->usrlibs>0)
dirnum=sbbs->usrdir[sbbs->curlib][sbbs->curdir[sbbs->curlib]];
return(dirnum);
}
/**************************/
/* bbs Object Methods */
/**************************/
static JSBool
js_menu(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
{
JSString* str;
sbbs_t* sbbs;
if((sbbs=(sbbs_t*)JS_GetContextPrivate(cx))==NULL)
return(JS_FALSE);
str = JS_ValueToString(cx, argv[0]);
if (!str)
return(JS_FALSE);
rc=JS_SUSPENDREQUEST(cx);
JS_RESUMEREQUEST(cx, rc);
static JSBool
js_hangup(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
{
sbbs_t* sbbs;
if((sbbs=(sbbs_t*)JS_GetContextPrivate(cx))==NULL)
return(JS_FALSE);
rc=JS_SUSPENDREQUEST(cx);
JS_RESUMEREQUEST(cx, rc);
static JSBool
js_nodesync(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
{
sbbs_t* sbbs;
if((sbbs=(sbbs_t*)JS_GetContextPrivate(cx))==NULL)
return(JS_FALSE);
rc=JS_SUSPENDREQUEST(cx);