Newer
Older
/* Synchronet JavaScript "User" 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 2003 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"
#ifdef JAVASCRIPT
static const char* getprivate_failure = "line %d %s JS_GetPrivate failed";

rswindell
committed
typedef struct
{

rswindell
committed
scfg_t* cfg;
} private_t;
/* User Object Properites */
enum {
,USER_PROP_NAME
,USER_PROP_HANDLE
,USER_PROP_NOTE
,USER_PROP_COMP
,USER_PROP_COMMENT
,USER_PROP_NETMAIL

rswindell
committed
,USER_PROP_EMAIL /* READ ONLY */
,USER_PROP_ADDRESS
,USER_PROP_LOCATION
,USER_PROP_BIRTH
,USER_PROP_AGE /* READ ONLY */
,USER_PROP_MODEM
,USER_PROP_LASTON
,USER_PROP_FIRSTON
,USER_PROP_EXPIRE
,USER_PROP_PWMOD
,USER_PROP_LOGONS
,USER_PROP_LTODAY
,USER_PROP_TIMEON
,USER_PROP_TEXTRA
,USER_PROP_TTODAY
,USER_PROP_TLAST
,USER_PROP_POSTS
,USER_PROP_EMAILS
,USER_PROP_FBACKS
,USER_PROP_ETODAY
,USER_PROP_PTODAY
,USER_PROP_MAIL_WAITING
,USER_PROP_MAIL_PENDING
,USER_PROP_ULB
,USER_PROP_ULS
,USER_PROP_DLB
,USER_PROP_DLS
,USER_PROP_CDT
,USER_PROP_MIN
,USER_PROP_LEVEL
,USER_PROP_FLAGS1
,USER_PROP_FLAGS2
,USER_PROP_FLAGS3
,USER_PROP_FLAGS4
,USER_PROP_EXEMPT
,USER_PROP_REST
,USER_PROP_ROWS
,USER_PROP_SEX
,USER_PROP_MISC
,USER_PROP_LEECH
,USER_PROP_CURSUB
,USER_PROP_CURDIR
,USER_PROP_CURXTRN
,USER_PROP_FREECDT
,USER_PROP_XEDIT
,USER_PROP_SHELL
,USER_PROP_QWK
,USER_PROP_TMPEXT
,USER_PROP_CHAT
,USER_PROP_NS_TIME
,USER_PROP_PROT
,USER_PROP_LOGONTIME
,USER_PROP_TIMEPERCALL
,USER_PROP_TIMEPERDAY
,USER_PROP_CALLSPERDAY
,USER_PROP_LINESPERMSG
,USER_PROP_EMAILPERDAY
,USER_PROP_POSTSPERDAY
,USER_PROP_FREECDTPERDAY
static JSBool js_user_get(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
char* s=NULL;
char tmp[128];
ulong val=0;

rswindell
committed
private_t* p;
if((p=(private_t*)JS_GetPrivate(cx,obj))==NULL) {
JS_ReportError(cx,getprivate_failure,WHERE);
return(JS_FALSE);
user.number=p->usernumber;
getuserdat(p->cfg,&user);
tiny = JSVAL_TO_INT(id);
switch(tiny) {
case USER_PROP_NUMBER:
val=user.number;
break;
case USER_PROP_ALIAS:
break;
case USER_PROP_NAME:
break;
case USER_PROP_HANDLE:
break;
case USER_PROP_NOTE:
break;
case USER_PROP_COMP:
break;
case USER_PROP_COMMENT:
break;
case USER_PROP_NETMAIL:

rswindell
committed
case USER_PROP_EMAIL:
s=usermailaddr(p->cfg, tmp
,p->cfg->inetmail_misc&NMAIL_ALIAS ? user.alias : user.name);
break;
case USER_PROP_ADDRESS:
break;
case USER_PROP_LOCATION:
break;
case USER_PROP_ZIPCODE:
break;
case USER_PROP_PASS:
break;
case USER_PROP_PHONE:
break;
case USER_PROP_BIRTH:
break;
case USER_PROP_AGE:
break;
case USER_PROP_MODEM:
break;
case USER_PROP_LASTON:
break;
case USER_PROP_FIRSTON:
break;
case USER_PROP_EXPIRE:
break;
case USER_PROP_PWMOD:
break;
case USER_PROP_LOGONS:
break;
case USER_PROP_LTODAY:
break;
case USER_PROP_TIMEON:
break;
case USER_PROP_TEXTRA:
break;
case USER_PROP_TTODAY:
break;
case USER_PROP_TLAST:
break;
case USER_PROP_POSTS:
break;
case USER_PROP_EMAILS:
break;
case USER_PROP_FBACKS:
break;
case USER_PROP_ETODAY:
break;
case USER_PROP_PTODAY:
break;
case USER_PROP_ULB:
break;
case USER_PROP_ULS:
break;
case USER_PROP_DLB:
break;
case USER_PROP_DLS:
break;
case USER_PROP_CDT:
break;
case USER_PROP_MIN:
break;
case USER_PROP_LEVEL:
break;
case USER_PROP_FLAGS1:
break;
case USER_PROP_FLAGS2:
break;
case USER_PROP_FLAGS3:
break;
case USER_PROP_FLAGS4:
break;
case USER_PROP_EXEMPT:
break;
case USER_PROP_REST:
break;
case USER_PROP_ROWS:
break;
case USER_PROP_SEX:
s=tmp;
break;
break;
case USER_PROP_LEECH:
break;
case USER_PROP_CURSUB:
break;
case USER_PROP_CURDIR:
case USER_PROP_CURXTRN:
s=user.curxtrn;
break;
case USER_PROP_FREECDT:
break;
case USER_PROP_XEDIT:

rswindell
committed
if(user.xedit>0 && user.xedit<=p->cfg->total_xedits)
s=p->cfg->xedit[user.xedit-1]->code;
else
s=""; /* internal editor */
break;
case USER_PROP_SHELL:

rswindell
committed
s=p->cfg->shell[user.shell]->code;
case USER_PROP_QWK:
break;
case USER_PROP_TMPEXT:
break;
case USER_PROP_CHAT:
break;
case USER_PROP_NS_TIME:
val=user.ns_time;
break;
case USER_PROP_PROT:
s=tmp;
break;
case USER_PROP_LOGONTIME:
val=user.logontime;
break;
case USER_PROP_TIMEPERCALL:
val=p->cfg->level_timepercall[user.level];
break;
case USER_PROP_TIMEPERDAY:
val=p->cfg->level_timeperday[user.level];
break;
case USER_PROP_CALLSPERDAY:
val=p->cfg->level_callsperday[user.level];
break;
case USER_PROP_LINESPERMSG:
val=p->cfg->level_linespermsg[user.level];
break;
case USER_PROP_POSTSPERDAY:
val=p->cfg->level_postsperday[user.level];
break;
case USER_PROP_EMAILPERDAY:
val=p->cfg->level_emailperday[user.level];
break;
case USER_PROP_FREECDTPERDAY:
val=p->cfg->level_freecdtperday[user.level];
break;
case USER_PROP_MAIL_WAITING:
val=getmail(p->cfg,user.number,/* sent? */FALSE);
break;
case USER_PROP_MAIL_PENDING:
val=getmail(p->cfg,user.number,/* sent? */TRUE);
break;
default:
/* This must not set vp in order for child objects to work (stats and security) */
return(JS_TRUE);
if(s!=NULL) {
if((js_str=JS_NewStringCopyZ(cx, s))==NULL)
return(JS_FALSE);
*vp = STRING_TO_JSVAL(js_str);
} else
JS_NewNumberValue(cx,val,vp);
return(JS_TRUE);
static JSBool js_user_set(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
char* str;
char tmp[64];
JSString* js_str;

rswindell
committed
private_t* p;
if((p=(private_t*)JS_GetPrivate(cx,obj))==NULL) {
JS_ReportError(cx,getprivate_failure,WHERE);
return(JS_FALSE);
if((js_str=JS_ValueToString(cx,*vp))==NULL)
return(JS_FALSE);
if((str=JS_GetStringBytes(js_str))==NULL)
return(JS_FALSE);
tiny = JSVAL_TO_INT(id);
switch(tiny) {
JS_ValueToInt32(cx, *vp, (jsint*)&p->usernumber);
case USER_PROP_ALIAS:
/* update USER.DAT */
putuserrec(p->cfg,p->usernumber,U_ALIAS,LEN_ALIAS,str);
/* update NAME.DAT */
getuserrec(p->cfg,p->usernumber,U_MISC,8,tmp);
usermisc=ahtoul(tmp);
if(!(usermisc&DELETED))
putusername(p->cfg,p->usernumber,str);
break;
case USER_PROP_NAME:
putuserrec(p->cfg,p->usernumber,U_NAME,LEN_NAME,str);
break;
case USER_PROP_HANDLE:
putuserrec(p->cfg,p->usernumber,U_HANDLE,LEN_HANDLE,str);
break;
case USER_PROP_NOTE:
putuserrec(p->cfg,p->usernumber,U_NOTE,LEN_NOTE,str);
break;
case USER_PROP_COMP:
putuserrec(p->cfg,p->usernumber,U_COMP,LEN_COMP,str);
break;
case USER_PROP_COMMENT:
putuserrec(p->cfg,p->usernumber,U_COMMENT,LEN_COMMENT,str);
break;
case USER_PROP_NETMAIL:
putuserrec(p->cfg,p->usernumber,U_NETMAIL,LEN_NETMAIL,str);
break;
case USER_PROP_ADDRESS:
putuserrec(p->cfg,p->usernumber,U_ADDRESS,LEN_ADDRESS,str);
break;
case USER_PROP_LOCATION:
putuserrec(p->cfg,p->usernumber,U_LOCATION,LEN_LOCATION,str);
break;
case USER_PROP_ZIPCODE:
putuserrec(p->cfg,p->usernumber,U_ZIPCODE,LEN_ZIPCODE,str);
break;
case USER_PROP_PHONE:
putuserrec(p->cfg,p->usernumber,U_PHONE,LEN_PHONE,str);
break;
case USER_PROP_BIRTH:
putuserrec(p->cfg,p->usernumber,U_BIRTH,LEN_BIRTH,str);
break;
case USER_PROP_MODEM:
putuserrec(p->cfg,p->usernumber,U_MODEM,LEN_MODEM,str);
break;
case USER_PROP_ROWS:
putuserrec(p->cfg,p->usernumber,U_ROWS,0,str); /* base 10 */
break;
case USER_PROP_SEX:
putuserrec(p->cfg,p->usernumber,U_SEX,0,strupr(str)); /* single char */
break;
case USER_PROP_CURSUB:
putuserrec(p->cfg,p->usernumber,U_CURSUB,0,strupr(str));
break;
case USER_PROP_CURDIR:
putuserrec(p->cfg,p->usernumber,U_CURDIR,0,strupr(str));
case USER_PROP_CURXTRN:
putuserrec(p->cfg,p->usernumber,U_CURXTRN,0,strupr(str));
break;
case USER_PROP_XEDIT:
putuserrec(p->cfg,p->usernumber,U_XEDIT,0,strupr(str));
break;
case USER_PROP_SHELL:
putuserrec(p->cfg,p->usernumber,U_COMP,0,strupr(str));
break;
case USER_PROP_MISC:
JS_ValueToInt32(cx,*vp,&val);
putuserrec(p->cfg,p->usernumber,U_MISC,0,ultoa(val,tmp,16));
break;
case USER_PROP_QWK:
JS_ValueToInt32(cx,*vp,&val);
putuserrec(p->cfg,p->usernumber,U_QWK,0,ultoa(val,tmp,16));
break;
case USER_PROP_CHAT:
JS_ValueToInt32(cx,*vp,&val);
putuserrec(p->cfg,p->usernumber,U_CHAT,0,ultoa(val,tmp,16));
break;
case USER_PROP_TMPEXT:
putuserrec(p->cfg,p->usernumber,U_TMPEXT,0,str);
break;
case USER_PROP_NS_TIME:
JS_ValueToInt32(cx,*vp,&val);
putuserrec(p->cfg,p->usernumber,U_NS_TIME,0,ultoa(val,tmp,16));
break;
case USER_PROP_PROT:
putuserrec(p->cfg,p->usernumber,U_PROT,0,strupr(str)); /* single char */
case USER_PROP_LOGONTIME:
JS_ValueToInt32(cx,*vp,&val);
putuserrec(p->cfg,p->usernumber,U_LOGONTIME,0,ultoa(val,tmp,16));
/* security properties*/
case USER_PROP_PASS:
putuserrec(p->cfg,p->usernumber,U_PASS,LEN_PASS,strupr(str));
break;
case USER_PROP_PWMOD:
JS_ValueToInt32(cx,*vp,&val);
putuserrec(p->cfg,p->usernumber,U_PWMOD,0,ultoa(val,tmp,16));
break;
case USER_PROP_LEVEL:
putuserrec(p->cfg,p->usernumber,U_LEVEL,0,str);
break;
case USER_PROP_FLAGS1:
JS_ValueToInt32(cx,*vp,&val);
putuserrec(p->cfg,p->usernumber,U_FLAGS1,0,ultoa(val,tmp,16));
break;
case USER_PROP_FLAGS2:
JS_ValueToInt32(cx,*vp,&val);
putuserrec(p->cfg,p->usernumber,U_FLAGS2,0,ultoa(val,tmp,16));
break;
case USER_PROP_FLAGS3:
JS_ValueToInt32(cx,*vp,&val);
putuserrec(p->cfg,p->usernumber,U_FLAGS3,0,ultoa(val,tmp,16));
break;
case USER_PROP_FLAGS4:
JS_ValueToInt32(cx,*vp,&val);
putuserrec(p->cfg,p->usernumber,U_FLAGS4,0,ultoa(val,tmp,16));
break;
case USER_PROP_EXEMPT:
JS_ValueToInt32(cx,*vp,&val);
putuserrec(p->cfg,p->usernumber,U_EXEMPT,0,ultoa(val,tmp,16));
break;
case USER_PROP_REST:
JS_ValueToInt32(cx,*vp,&val);
putuserrec(p->cfg,p->usernumber,U_REST,0,ultoa(val,tmp,16));
break;
case USER_PROP_CDT:
putuserrec(p->cfg,p->usernumber,U_CDT,0,str);
break;
case USER_PROP_FREECDT:
putuserrec(p->cfg,p->usernumber,U_FREECDT,0,str);
break;
case USER_PROP_MIN:
putuserrec(p->cfg,p->usernumber,U_MIN,0,str);
break;
case USER_PROP_TEXTRA:
putuserrec(p->cfg,p->usernumber,U_TEXTRA,0,str);
break;
case USER_PROP_EXPIRE:
JS_ValueToInt32(cx,*vp,&val);
putuserrec(p->cfg,p->usernumber,U_EXPIRE,0,ultoa(val,tmp,16));
return(JS_TRUE);
#define USER_PROP_FLAGS JSPROP_ENUMERATE
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
static jsSyncPropertySpec js_user_properties[] = {
/* name ,tinyid ,flags, ver */
{ "number" ,USER_PROP_NUMBER ,USER_PROP_FLAGS, 310},
{ "alias" ,USER_PROP_ALIAS ,USER_PROP_FLAGS, 310},
{ "name" ,USER_PROP_NAME ,USER_PROP_FLAGS, 310},
{ "handle" ,USER_PROP_HANDLE ,USER_PROP_FLAGS, 310},
{ "ip_address" ,USER_PROP_NOTE ,USER_PROP_FLAGS, 310},
{ "note" ,USER_PROP_NOTE ,USER_PROP_FLAGS, 310},
{ "host_name" ,USER_PROP_COMP ,USER_PROP_FLAGS, 310},
{ "computer" ,USER_PROP_COMP ,USER_PROP_FLAGS, 310},
{ "comment" ,USER_PROP_COMMENT ,USER_PROP_FLAGS, 310},
{ "netmail" ,USER_PROP_NETMAIL ,USER_PROP_FLAGS, 310},
{ "email" ,USER_PROP_EMAIL ,USER_PROP_FLAGS|JSPROP_READONLY, 310},
{ "address" ,USER_PROP_ADDRESS ,USER_PROP_FLAGS, 310},
{ "location" ,USER_PROP_LOCATION ,USER_PROP_FLAGS, 310},
{ "zipcode" ,USER_PROP_ZIPCODE ,USER_PROP_FLAGS, 310},
{ "phone" ,USER_PROP_PHONE ,USER_PROP_FLAGS, 310},
{ "birthdate" ,USER_PROP_BIRTH ,USER_PROP_FLAGS, 310},
{ "age" ,USER_PROP_AGE ,USER_PROP_FLAGS|JSPROP_READONLY, 310},
{ "connection" ,USER_PROP_MODEM ,USER_PROP_FLAGS, 310},
{ "modem" ,USER_PROP_MODEM ,USER_PROP_FLAGS, 310},
{ "screen_rows" ,USER_PROP_ROWS ,USER_PROP_FLAGS, 310},
{ "gender" ,USER_PROP_SEX ,USER_PROP_FLAGS, 310},
{ "cursub" ,USER_PROP_CURSUB ,USER_PROP_FLAGS, 310},
{ "curdir" ,USER_PROP_CURDIR ,USER_PROP_FLAGS, 310},
{ "curxtrn" ,USER_PROP_CURXTRN ,USER_PROP_FLAGS, 310},
{ "editor" ,USER_PROP_XEDIT ,USER_PROP_FLAGS, 310},
{ "command_shell" ,USER_PROP_SHELL ,USER_PROP_FLAGS, 310},
{ "settings" ,USER_PROP_MISC ,USER_PROP_FLAGS, 310},
{ "qwk_settings" ,USER_PROP_QWK ,USER_PROP_FLAGS, 310},
{ "chat_settings" ,USER_PROP_CHAT ,USER_PROP_FLAGS, 310},
{ "temp_file_ext" ,USER_PROP_TMPEXT ,USER_PROP_FLAGS, 310},
{ "new_file_time" ,USER_PROP_NS_TIME ,USER_PROP_FLAGS, 311},
{ "newscan_date" ,USER_PROP_NS_TIME ,0, /* Alias */ 310},
{ "download_protocol" ,USER_PROP_PROT ,USER_PROP_FLAGS, 310},
{ "logontime" ,USER_PROP_LOGONTIME ,USER_PROP_FLAGS, 310},
#ifdef _DEBUG
static char* user_prop_desc[] = {
"record number (1-based)"
,"real name"
,"chat handle"
,"IP address last logged on from"
,"AKA ip_address"
,"host name last logged on from"
,"AKA host_name"
,"sysop's comment"
,"local Internet e-mail address - <small>READ ONLY</small>"
,"street address"
,"location (city, state)"
,"zip/postal code"
,"phone number"
,"birth date"
,"calculated age in years - <small>READ ONLY</small>"
,"AKA connection"
,"terminal rows (lines)"
,"current message sub-board"
,"current file directory"
,"current external program being run"
,"external message editor"
,"command shell"
,"settings bitfield - see <tt>USER_*</tt> in <tt>sbbsdefs.js</tt> for bit definitions"
,"QWK packet settings bitfield - see <tt>QWK_*</tt> in <tt>sbbsdefs.js</tt> for bit definitions"
,"chat settings bitfield - see <tt>CHAT_*</tt> in <tt>sbbsdefs.js</tt> for bit definitions"
,"temporary file type (extension)"
,"new file scan date/time (time_t format)"
,"file transfer protocol (command key)"
,"logon time (time_t format)"
/* user.security */
static jsSyncPropertySpec js_user_security_properties[] = {
/* name ,tinyid ,flags, ver */
{ "password" ,USER_PROP_PASS ,USER_PROP_FLAGS, 310 },
{ "password_date" ,USER_PROP_PWMOD ,USER_PROP_FLAGS, 310 },
{ "level" ,USER_PROP_LEVEL ,USER_PROP_FLAGS, 310 },
{ "flags1" ,USER_PROP_FLAGS1 ,USER_PROP_FLAGS, 310 },
{ "flags2" ,USER_PROP_FLAGS2 ,USER_PROP_FLAGS, 310 },
{ "flags3" ,USER_PROP_FLAGS3 ,USER_PROP_FLAGS, 310 },
{ "flags4" ,USER_PROP_FLAGS4 ,USER_PROP_FLAGS, 310 },
{ "exemptions" ,USER_PROP_EXEMPT ,USER_PROP_FLAGS, 310 },
{ "restrictions" ,USER_PROP_REST ,USER_PROP_FLAGS, 310 },
{ "credits" ,USER_PROP_CDT ,USER_PROP_FLAGS, 310 },
{ "free_credits" ,USER_PROP_FREECDT ,USER_PROP_FLAGS, 310 },
{ "minutes" ,USER_PROP_MIN ,USER_PROP_FLAGS, 310 },
{ "extra_time" ,USER_PROP_TEXTRA ,USER_PROP_FLAGS, 310 },
{ "expiration_date" ,USER_PROP_EXPIRE ,USER_PROP_FLAGS, 310 },
#ifdef _DEBUG
static char* user_security_prop_desc[] = {
"password"
,"date password last modified (time_t format)"
,"security level (0-99)"
,"flag set #1 (bitfield)"
,"flag set #2 (bitfield)"
,"flag set #3 (bitfield)"
,"flag set #4 (bitfield)"
,"exemption flags (bitfield)"
,"restriction flags (bitfield)"
,"credits"
,"free credits (for today only)"
,"extra minutes (time bank)"
,"extra minutes (for today only)"
,"expiration date/time (time_t format)"
#undef USER_PROP_FLAGS
#define USER_PROP_FLAGS JSPROP_ENUMERATE|JSPROP_READONLY
/* user.limits: These should be READ ONLY by nature */
static jsSyncPropertySpec js_user_limits_properties[] = {
/* name ,tinyid ,flags, ver */
{ "time_per_logon" ,USER_PROP_TIMEPERCALL ,USER_PROP_FLAGS, 311 },
{ "time_per_day" ,USER_PROP_TIMEPERDAY ,USER_PROP_FLAGS, 311 },
{ "logons_per_day" ,USER_PROP_CALLSPERDAY ,USER_PROP_FLAGS, 311 },
{ "lines_per_message" ,USER_PROP_LINESPERMSG ,USER_PROP_FLAGS, 311 },
{ "email_per_day" ,USER_PROP_EMAILPERDAY ,USER_PROP_FLAGS, 311 },
{ "posts_per_day" ,USER_PROP_POSTSPERDAY ,USER_PROP_FLAGS, 311 },
{ "free_credits_per_day" ,USER_PROP_FREECDTPERDAY,USER_PROP_FLAGS, 311 },
{0}
};
#ifdef _DEBUG
static char* user_limits_prop_desc[] = {
"time (in minutes) per logon"
,"time (in minutes) per day"
,"logons per day"
,"lines per message (post or email)"
,"email sent per day"
,"messages posted per day"
,"free credits given per day"
,NULL
};
#endif
#undef USER_PROP_FLAGS
#define USER_PROP_FLAGS JSPROP_ENUMERATE|JSPROP_READONLY
/* user.stats: These should be READ ONLY by nature */
static jsSyncPropertySpec js_user_stats_properties[] = {
/* name ,tinyid ,flags, ver */
{ "laston_date" ,USER_PROP_LASTON ,USER_PROP_FLAGS, 310 },
{ "firston_date" ,USER_PROP_FIRSTON ,USER_PROP_FLAGS, 310 },
{ "total_logons" ,USER_PROP_LOGONS ,USER_PROP_FLAGS, 310 },
{ "logons_today" ,USER_PROP_LTODAY ,USER_PROP_FLAGS, 310 },
{ "total_timeon" ,USER_PROP_TIMEON ,USER_PROP_FLAGS, 310 },
{ "timeon_today" ,USER_PROP_TTODAY ,USER_PROP_FLAGS, 310 },
{ "timeon_last_logon" ,USER_PROP_TLAST ,USER_PROP_FLAGS, 310 },
{ "total_posts" ,USER_PROP_POSTS ,USER_PROP_FLAGS, 310 },
{ "total_emails" ,USER_PROP_EMAILS ,USER_PROP_FLAGS, 310 },
{ "total_feedbacks" ,USER_PROP_FBACKS ,USER_PROP_FLAGS, 310 },
{ "email_today" ,USER_PROP_ETODAY ,USER_PROP_FLAGS, 310 },
{ "posts_today" ,USER_PROP_PTODAY ,USER_PROP_FLAGS, 310 },
{ "bytes_uploaded" ,USER_PROP_ULB ,USER_PROP_FLAGS, 310 },
{ "files_uploaded" ,USER_PROP_ULS ,USER_PROP_FLAGS, 310 },
{ "bytes_downloaded" ,USER_PROP_DLB ,USER_PROP_FLAGS, 310 },
{ "files_downloaded" ,USER_PROP_DLS ,USER_PROP_FLAGS, 310 },
{ "leech_attempts" ,USER_PROP_LEECH ,USER_PROP_FLAGS, 310 },
{ "mail_waiting" ,USER_PROP_MAIL_WAITING ,USER_PROP_FLAGS, 312 },
{ "mail_pending" ,USER_PROP_MAIL_PENDING ,USER_PROP_FLAGS, 312 },
#ifdef _DEBUG
static char* user_stats_prop_desc[] = {
"date of previous logon"
,"date of first logon"
,"total number of logons"
,"total logons today"
,"total time used (in minutes)"
,"time used today"
,"time used last session"
,"total messages posted"
,"total e-mails sent"
,"total feedback messages sent"
,"e-mail sent today"
,"messages posted today"
,"total bytes uploaded"
,"total files uploaded"
,"total bytes downloaded"
,"total files downloaded"
,"suspected leech downloads"
,NULL
};
#endif

rswindell
committed
static void js_user_finalize(JSContext *cx, JSObject *obj)
{
private_t* p;
p=(private_t*)JS_GetPrivate(cx,obj);

rswindell
committed
free(p);
p=NULL;
JS_SetPrivate(cx,obj,p);
}
static JSBool
js_chk_ar(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
{

rswindell
committed
JSString* js_str;

rswindell
committed
private_t* p;
if((p=(private_t*)JS_GetPrivate(cx,obj))==NULL)
return JS_FALSE;
if((js_str=JS_ValueToString(cx, argv[0]))==NULL)
return JS_FALSE;
ar = arstr(NULL,JS_GetStringBytes(js_str),p->cfg);
user.number=p->usernumber;
getuserdat(p->cfg,&user);
*rval = BOOLEAN_TO_JSVAL(chk_ar(p->cfg,ar,&user));

rswindell
committed
if(ar!=NULL && ar!=nular)
free(ar);
return(JS_TRUE);
}
static jsSyncMethodSpec js_user_functions[] = {
{"compare_ars", js_chk_ar, 1, JSTYPE_BOOLEAN, JSDOCSTR("string ars")
,JSDOCSTR("Verify user meets access requirements string")
,310

rswindell
committed
{0}
};
static JSClass js_user_class = {
"User" /* name */
,JSCLASS_HAS_PRIVATE /* flags */
,JS_PropertyStub /* addProperty */
,JS_PropertyStub /* delProperty */
,js_user_get /* getProperty */
,js_user_set /* setProperty */
,JS_EnumerateStub /* enumerate */
,JS_ResolveStub /* resolve */
,JS_ConvertStub /* convert */

rswindell
committed
,js_user_finalize /* finalize */
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
static JSClass js_user_stats_class = {
"UserStats" /* name */
,JSCLASS_HAS_PRIVATE /* flags */
,JS_PropertyStub /* addProperty */
,JS_PropertyStub /* delProperty */
,js_user_get /* getProperty */
,js_user_set /* setProperty */
,JS_EnumerateStub /* enumerate */
,JS_ResolveStub /* resolve */
,JS_ConvertStub /* convert */
,JS_FinalizeStub /* finalize */
};
static JSClass js_user_security_class = {
"UserSecurity" /* name */
,JSCLASS_HAS_PRIVATE /* flags */
,JS_PropertyStub /* addProperty */
,JS_PropertyStub /* delProperty */
,js_user_get /* getProperty */
,js_user_set /* setProperty */
,JS_EnumerateStub /* enumerate */
,JS_ResolveStub /* resolve */
,JS_ConvertStub /* convert */
,JS_FinalizeStub /* finalize */
};
static JSClass js_user_limits_class = {
"UserLimits" /* name */
,JSCLASS_HAS_PRIVATE /* flags */
,JS_PropertyStub /* addProperty */
,JS_PropertyStub /* delProperty */
,js_user_get /* getProperty */
,js_user_set /* setProperty */
,JS_EnumerateStub /* enumerate */
,JS_ResolveStub /* resolve */
,JS_ConvertStub /* convert */
,JS_FinalizeStub /* finalize */
};
/* User Constructor (creates instance of user class) */
static JSBool
js_user_constructor(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
{
int i;
JSObject* statsobj;
JSObject* securityobj;
JSObject* limitsobj;
JS_ValueToInt32(cx,argv[0],&val);
user.number=(ushort)val;
if(user.number!=0 && (i=getuserdat(scfg,&user))!=0) {
JS_ReportError(cx,"Error %d reading user number %d",i,val);
/* user.stats */
if((statsobj=JS_DefineObject(cx, obj, "stats"
,&js_user_stats_class, NULL, JSPROP_ENUMERATE|JSPROP_READONLY))==NULL)
return(JS_FALSE);
if(!js_DefineSyncProperties(cx, statsobj, js_user_stats_properties))
return(JS_FALSE);
/* user.security */
if((securityobj=JS_DefineObject(cx, obj, "security"
,&js_user_security_class, NULL, JSPROP_ENUMERATE|JSPROP_READONLY))==NULL)
return(JS_FALSE);
if(!js_DefineSyncProperties(cx, securityobj, js_user_security_properties))
return(JS_FALSE);
/* user.limits */
if((limitsobj=JS_DefineObject(cx, obj, "limits"
,&js_user_limits_class, NULL, JSPROP_ENUMERATE|JSPROP_READONLY))==NULL)
return(JS_FALSE);
if(!js_DefineSyncProperties(cx, limitsobj, js_user_limits_properties))
return(JS_FALSE);
/* other user properties */
if(!js_DefineSyncProperties(cx, obj, js_user_properties))
return(JS_FALSE);
if((p=(private_t*)malloc(sizeof(private_t)))==NULL)
return(JS_FALSE);
p->cfg = scfg;
p->usernumber = user.number;
JS_SetPrivate(cx, obj, p); /* Must do this before calling js_DefineSyncMethods() */
if(!js_DefineSyncMethods(cx, obj, js_user_functions, FALSE)) {
JS_SetPrivate(cx, obj, NULL);
free(p);
return(JS_FALSE);
}
JS_SetPrivate(cx, statsobj, p);
JS_SetPrivate(cx, securityobj, p);
JS_SetPrivate(cx, limitsobj, p);
return(JS_TRUE);
}
JSObject* DLLCALL js_CreateUserClass(JSContext* cx, JSObject* parent, scfg_t* cfg)
{
JSObject* userclass;
scfg = cfg;
userclass = JS_InitClass(cx, parent, NULL
,&js_user_class
,js_user_constructor
,1 /* number of constructor args */
,NULL /* props, defined in constructor */
,NULL /* funcs, defined in constructor */
,NULL,NULL);
return(userclass);
}
JSObject* DLLCALL js_CreateUserObject(JSContext* cx, JSObject* parent, scfg_t* cfg, char* name
, uint usernumber)
{
JSObject* userobj;
JSObject* statsobj;
JSObject* limitsobj;

rswindell
committed
private_t* p;
jsval val;

rswindell
committed
if(name==NULL)
userobj = JS_NewObject(cx, &js_user_class, NULL, parent);
else if(JS_GetProperty(cx,parent,name,&val) && val!=JSVAL_VOID)
userobj = JSVAL_TO_OBJECT(val); /* Return existing user object */
else
userobj = JS_DefineObject(cx, parent, name, &js_user_class
, NULL, JSPROP_ENUMERATE|JSPROP_READONLY);
if(userobj==NULL)
return(NULL);
if((p=(private_t*)malloc(sizeof(private_t)))==NULL)
return(NULL);
p->cfg = cfg;
p->usernumber = usernumber;

rswindell
committed
JS_SetPrivate(cx, userobj, p);
if(!js_DefineSyncProperties(cx, userobj, js_user_properties)) {
free(p);
return(NULL);
}
js_DescribeSyncObject(cx,userobj
,"Instance of <i>User</i> class, representing current user online"
,310);
js_DescribeSyncConstructor(cx,userobj
,"To create a new user object: <tt>var u = new User(<i>number</i>)</tt>");
js_CreateArrayOfStrings(cx, userobj
,"_property_desc_list", user_prop_desc, JSPROP_READONLY);
js_DefineSyncMethods(cx, userobj, js_user_functions, FALSE);