atcodes.cpp 39.1 KB
Newer Older
1
/* Synchronet "@code" functions */
2
// vi: tabstop=4
3
4
5
6
7
8
9

/* $Id$ */

/****************************************************************************
 * @format.tab-size 4		(Plain Text/Source Code File Header)			*
 * @format.use-tabs true	(see http://www.synchro.net/ptsc_hdr.html)		*
 *																			*
10
 * Copyright Rob Swindell - http://www.synchro.net/copyright.html			*
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
 *																			*
 * 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 "cmdshell.h"
39
40
#include "utf8.h"
#include "unicode.h"
41
#include "cp437defs.h"
42

43
44
45
46
47
48
49
#if defined(_WINSOCKAPI_)
	extern WSADATA WSAData;
	#define SOCKLIB_DESC WSAData.szDescription
#else
	#define	SOCKLIB_DESC NULL
#endif

50
51
52
/****************************************************************************/
/* Returns 0 if invalid @ code. Returns length of @ code if valid.          */
/****************************************************************************/
53
int sbbs_t::show_atcode(const char *instr)
54
{
55
	char	str[128],str2[128],*tp,*sp,*p;
56
    int     len;
57
	int		disp_len;
58
59
	bool	padded_left=false;
	bool	padded_right=false;
60
	bool	centered=false;
61
	bool	zero_padded=false;
rswindell's avatar
rswindell committed
62
	bool	truncated = true;
rswindell's avatar
rswindell committed
63
	bool	doubled = false;
64
	long	pmode = 0;
65
	const char *cp;
66

67
	SAFECOPY(str,instr);
68
69
70
	tp=strchr(str+1,'@');
	if(!tp)                 /* no terminating @ */
		return(0);
71
	sp=strchr(str+1,' ');
72
73
74
75
76
77
	if(sp && sp<tp)         /* space before terminating @ */
		return(0);
	len=(tp-str)+1;
	(*tp)=0;
	sp=(str+1);

78
	disp_len=len;
79
80
81
	if(strchr(sp, ':') != NULL)
		p = NULL;
	else if((p=strstr(sp,"-L"))!=NULL)
82
83
84
		padded_left=true;
	else if((p=strstr(sp,"-R"))!=NULL)
		padded_right=true;
85
86
	else if((p=strstr(sp,"-C"))!=NULL)
		centered=true;
rswindell's avatar
rswindell committed
87
	else if((p=strstr(sp,"-W"))!=NULL)	/* wide */
rswindell's avatar
rswindell committed
88
		doubled=true;
89
90
	else if((p=strstr(sp,"-Z"))!=NULL)
		zero_padded=true;
rswindell's avatar
rswindell committed
91
	else if((p=strstr(sp,"->"))!=NULL)	/* wrap */
rswindell's avatar
rswindell committed
92
		truncated = false;
93
	if(p!=NULL) {
rswindell's avatar
rswindell committed
94
95
96
		char* lp = p + 2;
		if(*lp && isdigit(*lp))
			disp_len=atoi(lp);
97
		*p=0;
98
	}
99

100
	cp = atcode(sp, str2, sizeof(str2), &pmode);
101
	if(cp==NULL)
102
103
		return(0);

rswindell's avatar
rswindell committed
104
105
106
107
108
109
110
111
112
113
114
	if(p==NULL || truncated == false)
		disp_len = strlen(cp);

	if(truncated) {
		if(column + disp_len > cols - 1) {
			if(column >= cols - 1)
				disp_len = 0;
			else
				disp_len = (cols - 1) - column;
		}
	}
115
116
117
118
119
120
	if(pmode & P_UTF8) {
		if(term_supports(UTF8))
			disp_len += strlen(cp) - utf8_str_total_width(cp);
		else
			disp_len += strlen(cp) - utf8_str_count_width(cp, /* min: */1, /* max: */2);
	}
121
	if(padded_left)
122
		bprintf(pmode, "%-*.*s",disp_len,disp_len,cp);
123
	else if(padded_right)
124
		bprintf(pmode, "%*.*s",disp_len,disp_len,cp);
125
	else if(centered) {
rswindell's avatar
rswindell committed
126
		int vlen = strlen(cp);
127
128
		if(vlen < disp_len) {
			int left = (disp_len - vlen) / 2;
129
			bprintf(pmode, "%*s%-*s", left, "", disp_len - left, cp);
130
		} else
131
			bprintf(pmode, "%.*s", disp_len, cp);
rswindell's avatar
rswindell committed
132
133
	} else if(doubled) {
		wide(cp);
134
	} else if(zero_padded) {
135
136
		int vlen = strlen(cp);
		if(vlen < disp_len)
137
			bprintf(pmode, "%-.*s%s", (int)(disp_len - strlen(cp)), "0000000000", cp);
138
		else
139
			bprintf(pmode, "%.*s", disp_len, cp);
140
	} else
141
		bprintf(pmode, "%.*s", disp_len, cp);
142
143
144
145

	return(len);
}

146
const char* sbbs_t::atcode(char* sp, char* str, size_t maxlen, long* pmode)
147
{
148
	char*	tp = NULL;
149
	uint	i;
150
151
	uint	ugrp;
	uint	usub;
152
153
154
155
156
157
158
	long	l;
    stats_t stats;
    node_t  node;
	struct	tm tm;

	str[0]=0;

159
160
	if(strncmp(sp, "U+", 2) == 0) {	// UNICODE
		enum unicode_codepoint codepoint = (enum unicode_codepoint)strtoul(sp + 2, &tp, 16);
rswindell's avatar
rswindell committed
161
		if(tp == NULL || *tp == 0)
162
			outchar(codepoint, unicode_to_cp437(codepoint));
rswindell's avatar
rswindell committed
163
164
		else if(*tp == ':')
			outchar(codepoint, tp + 1);
165
166
		else {
			char fallback = (char)strtoul(tp + 1, NULL, 16);
rswindell's avatar
rswindell committed
167
			if(*tp == ',')
168
169
170
171
172
173
174
175
176
177
178
179
				outchar(codepoint, fallback);
			else if(*tp == '!') {
				char ch = unicode_to_cp437(codepoint);
				if(ch != 0)
					fallback = ch;
				outchar(codepoint, fallback);
			}
			else return NULL; // Invalid @-code
		}
		return nulstr;
	}

180
181
182
183
184
	if(strcmp(sp, "CHECKMARK") == 0) {
		outchar(UNICODE_CHECK_MARK, CP437_CHECK_MARK);
		return nulstr;
	}

rswindell's avatar
rswindell committed
185
186
187
188
	if(strcmp(sp, "ELLIPSIS") == 0) {
		outchar(UNICODE_HORIZONTAL_ELLIPSIS, "...");
		return nulstr;
	}
189
	if(strcmp(sp, "COPY") == 0) {
rswindell's avatar
rswindell committed
190
191
192
193
194
195
196
		outchar(UNICODE_COPYRIGHT_SIGN, "(C)");
		return nulstr;
	}
	if(strcmp(sp, "SOUNDCOPY") == 0) {
		outchar(UNICODE_SOUND_RECORDING_COPYRIGHT, "(P)");
		return nulstr;
	}
rswindell's avatar
rswindell committed
197
198
199
200
	if(strcmp(sp, "REGISTERED") == 0) {
		outchar(UNICODE_REGISTERED_SIGN, "(R)");
		return nulstr;
	}
rswindell's avatar
rswindell committed
201
202
203
204
205
	if(strcmp(sp, "TRADEMARK") == 0) {
		outchar(UNICODE_TRADE_MARK_SIGN, "(TM)");
		return nulstr;
	}
	if(strcmp(sp, "DEGREE_C") == 0) {
rswindell's avatar
rswindell committed
206
		outchar(UNICODE_DEGREE_CELSIUS, "\xF8""C");
rswindell's avatar
rswindell committed
207
208
209
		return nulstr;
	}
	if(strcmp(sp, "DEGREE_F") == 0) {
rswindell's avatar
rswindell committed
210
		outchar(UNICODE_DEGREE_FAHRENHEIT, "\xF8""F");
rswindell's avatar
rswindell committed
211
212
213
		return nulstr;
	}

rswindell's avatar
rswindell committed
214
215
216
217
218
	if(strncmp(sp, "WIDE:", 5) == 0) {
		wide(sp + 5);
		return(nulstr);
	}

219
	if(!strcmp(sp,"VER"))
220
		return(VERSION);
221

222
	if(!strcmp(sp,"REV")) {
223
		safe_snprintf(str,maxlen,"%c",REVISION);
224
225
		return(str);
	}
226

227
	if(!strcmp(sp,"FULL_VER")) {
228
		safe_snprintf(str,maxlen,"%s%c%s",VERSION,REVISION,beta_version);
229
		truncsp(str);
230
#if defined(_DEBUG)
231
		strcat(str," Debug");
232
#endif
233
		return(str);
234
235
	}

236
	if(!strcmp(sp,"VER_NOTICE"))
237
		return(VERSION_NOTICE);
238

239
240
	if(!strcmp(sp,"OS_VER"))
		return(os_version(str));
241
242

#ifdef JAVASCRIPT
243
244
	if(!strcmp(sp,"JS_VER"))
		return((char *)JS_GetImplementationVersion());
245
246
#endif

247
248
	if(!strcmp(sp,"PLATFORM"))
		return(PLATFORM_DESC);
249

250
251
	if(!strcmp(sp,"COPYRIGHT"))
		return(COPYRIGHT_NOTICE);
252

253
	if(!strcmp(sp,"COMPILER")) {
254
255
256
		char compiler[32];
		DESCRIBE_COMPILER(compiler);
		strncpy(str, compiler, maxlen);
257
		return(str);
258
259
	}

260
	if(!strcmp(sp,"UPTIME")) {
261
		extern volatile time_t uptime;
262
263
264
265
		time_t up=0;
		now = time(NULL);
		if (uptime != 0 && now >= uptime)
			up = now-uptime;
266
267
		char   days[64]="";
		if((up/(24*60*60))>=2) {
268
	        sprintf(days,"%lu days ",(ulong)(up/(24L*60L*60L)));
269
270
			up%=(24*60*60);
		}
271
		safe_snprintf(str,maxlen,"%s%lu:%02lu"
272
	        ,days
273
274
			,(ulong)(up/(60L*60L))
			,(ulong)((up/60L)%60L)
275
			);
276
		return(str);
277
278
	}

279
	if(!strcmp(sp,"SERVED")) {
280
		extern volatile ulong served;
281
		safe_snprintf(str,maxlen,"%lu",served);
282
283
284
		return(str);
	}

285
	if(!strcmp(sp,"SOCKET_LIB"))
286
		return(socklib_version(str,SOCKLIB_DESC));
287

288
	if(!strcmp(sp,"MSG_LIB")) {
289
		safe_snprintf(str,maxlen,"SMBLIB %s",smb_lib_ver());
290
291
		return(str);
	}
292

293
294
	if(!strcmp(sp,"BBS") || !strcmp(sp,"BOARDNAME"))
		return(cfg.sys_name);
295

296
	if(!strcmp(sp,"BAUD") || !strcmp(sp,"BPS")) {
297
		safe_snprintf(str,maxlen,"%lu",cur_output_rate ? cur_output_rate : cur_rate);
298
299
		return(str);
	}
300

rswindell's avatar
rswindell committed
301
302
303
304
305
306
307
308
	if(!strcmp(sp,"COLS")) {
		safe_snprintf(str,maxlen,"%lu",cols);
		return(str);
	}
	if(!strcmp(sp,"ROWS")) {
		safe_snprintf(str,maxlen,"%lu",rows);
		return(str);
	}
309
310
311
312
313
	if(strcmp(sp,"TERM") == 0)
		return term_type();

	if(strcmp(sp,"CHARSET") == 0)
		return term_charset();
rswindell's avatar
rswindell committed
314

315
316
	if(!strcmp(sp,"CONN"))
		return(connection);
317

318
319
	if(!strcmp(sp,"SYSOP"))
		return(cfg.sys_op);
320

321
322
	if(!strcmp(sp,"LOCATION"))
		return(cfg.sys_location);
323

324
	if(strcmp(sp,"NODE") == 0 || strcmp(sp,"NN") == 0) {
325
		safe_snprintf(str,maxlen,"%u",cfg.node_num);
326
327
		return(str);
	}
328
	if(strcmp(sp, "TNODES") == 0 || strcmp(sp, "TNODE") == 0 || strcmp(sp, "TN") == 0) {
329
		safe_snprintf(str,maxlen,"%u",cfg.sys_nodes);
330
331
		return(str);
	}
332
333
334
335
336
337
338
339
	if(strcmp(sp, "ANODES") == 0 || strcmp(sp, "ANODE") == 0 || strcmp(sp, "AN") == 0) {
		safe_snprintf(str, maxlen, "%u", count_nodes(/* self: */true));
		return str;
	}
	if(strcmp(sp, "ONODES") == 0 || strcmp(sp, "ONODE") == 0 || strcmp(sp, "ON") == 0) {
		safe_snprintf(str, maxlen, "%u", count_nodes(/* self: */false));
		return str;
	}
340

rswindell's avatar
rswindell committed
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
	if(strcmp(sp, "PWDAYS") == 0) {
		if(cfg.sys_pwdays) {
			safe_snprintf(str, maxlen, "%u", cfg.sys_pwdays);
			return str;
		}
		return text[Unlimited];
	}

	if(strcmp(sp, "AUTODEL") == 0) {
		if(cfg.sys_autodel) {
			safe_snprintf(str, maxlen, "%u", cfg.sys_autodel);
			return str;
		}
		return text[Unlimited];
	}

357
358
359
360
361
362
	if(strcmp(sp, "PAGER") == 0)
		return (thisnode.misc&NODE_POFF) ? text[Off] : text[On];

	if(strcmp(sp, "ALERTS") == 0)
		return (thisnode.misc&NODE_AOFF) ? text[Off] : text[On];

363
364
365
	if(strcmp(sp, "SPLITP") == 0)
		return (useron.chat&CHAT_SPLITP) ? text[On] : text[Off];

366
367
	if(!strcmp(sp,"INETADDR"))
		return(cfg.sys_inetaddr);
368

369
370
371
	if(!strcmp(sp,"HOSTNAME"))
		return(startup->host_name);

372
	if(!strcmp(sp,"FIDOADDR")) {
373
		if(cfg.total_faddrs)
374
			return(smb_faddrtoa(&cfg.faddr[0],str));
375
		return(nulstr);
376
377
	}

378
	if(!strcmp(sp,"EMAILADDR"))
379
		return(usermailaddr(&cfg, str
380
			,cfg.inetmail_misc&NMAIL_ALIAS ? useron.alias : useron.name));
381

382
383
	if(!strcmp(sp,"QWKID"))
		return(cfg.sys_id);
384

385
	if(!strcmp(sp,"TIME") || !strcmp(sp,"SYSTIME")) {
386
		now=time(NULL);
387
		memset(&tm,0,sizeof(tm));
388
		localtime_r(&now,&tm);
389
		if(cfg.sys_misc&SM_MILITARY)
390
391
			safe_snprintf(str,maxlen,"%02d:%02d:%02d"
		        	,tm.tm_hour,tm.tm_min,tm.tm_sec);
392
		else
393
			safe_snprintf(str,maxlen,"%02d:%02d %s"
394
395
396
				,tm.tm_hour==0 ? 12
				: tm.tm_hour>12 ? tm.tm_hour-12
				: tm.tm_hour, tm.tm_min, tm.tm_hour>11 ? "pm":"am");
397
		return(str);
398
399
	}

rswindell's avatar
rswindell committed
400
401
402
	if(!strcmp(sp,"TIMEZONE"))
		return(smb_zonestr(sys_timezone(&cfg),str));

403
	if(!strcmp(sp,"DATE") || !strcmp(sp,"SYSDATE")) {
404
		return(unixtodstr(&cfg,time32(NULL),str));
405
	}
406

rswindell's avatar
rswindell committed
407
408
409
	if(!strcmp(sp,"DATETIME"))
		return(timestr(time(NULL)));

rswindell's avatar
rswindell committed
410
411
412
413
414
	if(!strcmp(sp,"DATETIMEZONE")) {
		char zone[32];
		safe_snprintf(str, maxlen, "%s %s", timestr(time(NULL)), smb_zonestr(sys_timezone(&cfg),zone));
		return str;
	}
415
416
417
418
	
	if(strcmp(sp, "DATEFMT") == 0) {
		return cfg.sys_misc&SM_EURODATE ? "DD/MM/YY" : "MM/DD/YY";
	}
rswindell's avatar
rswindell committed
419

420
	if(!strcmp(sp,"TMSG")) {
421
422
		l=0;
		for(i=0;i<cfg.total_subs;i++)
423
			l+=getposts(&cfg,i); 		/* l=total posts */
424
		safe_snprintf(str,maxlen,"%lu",l);
425
426
		return(str);
	}
427

428
	if(!strcmp(sp,"TUSER")) {
429
		safe_snprintf(str,maxlen,"%u",total_users(&cfg));
430
431
		return(str);
	}
432

433
	if(!strcmp(sp,"TFILE")) {
434
435
		l=0;
		for(i=0;i<cfg.total_dirs;i++)
436
			l+=getfiles(&cfg,i);
437
		safe_snprintf(str,maxlen,"%lu",l);
438
439
		return(str);
	}
440

441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
	if(strncmp(sp, "FILES:", 6) == 0) {	// Number of files in specified directory
		for(i = 0; i < cfg.total_dirs; i++) {
			if(stricmp(cfg.dir[i]->code, sp + 6) == 0) {
				safe_snprintf(str, maxlen, "%lu", (ulong)getfiles(&cfg, i));
				return str;
			}
		}
		return nulstr;
	}

	if(strcmp(sp, "FILES") == 0) {	// Number of files in current directory
		safe_snprintf(str, maxlen, "%lu", (ulong)getfiles(&cfg, usrdir[curlib][curdir[curlib]]));
		return str;
	}

456
	if(!strcmp(sp,"TCALLS") || !strcmp(sp,"NUMCALLS")) {
457
		getstats(&cfg,0,&stats);
458
		safe_snprintf(str,maxlen,"%lu", (ulong)stats.logons);
459
460
		return(str);
	}
461

462
	if(!strcmp(sp,"PREVON") || !strcmp(sp,"LASTCALLERNODE")
463
		|| !strcmp(sp,"LASTCALLERSYSTEM"))
464
		return(lastuseron);
465

rswindell's avatar
rswindell committed
466
	if(!strcmp(sp,"CLS") || !strcmp(sp,"CLEAR")) {
467
		CLS;
468
469
		return(nulstr);
	}
470

471
	if(!strcmp(sp,"PAUSE") || !strcmp(sp,"MORE")) {
472
		pause();
473
474
		return(nulstr);
	}
475

476
	if(!strcmp(sp,"RESETPAUSE")) {
477
		lncntr=0;
478
479
		return(nulstr);
	}
480

481
	if(!strcmp(sp,"NOPAUSE") || !strcmp(sp,"POFF")) {
482
		sys_status^=SS_PAUSEOFF;
483
484
		return(nulstr);
	}
485

486
	if(!strcmp(sp,"PON") || !strcmp(sp,"AUTOMORE")) {
487
		sys_status^=SS_PAUSEON;
488
489
		return(nulstr);
	}
490

rswindell's avatar
rswindell committed
491
492
493
494
495
496
497
	if(strncmp(sp, "FILL:", 5) == 0) {
		sp += 5;
		while(*sp && online && column < cols - 1)
			bputs(sp, P_TRUNCATE);
		return nulstr;
	}

498
499
500
501
502
	if(strncmp(sp, "POS:", 4) == 0) {	// PCBoard	(nn is 1 based)
		i = atoi(sp + 4);
		if(i >= 1)	// Convert to 0-based
			i--;
		for(l = i - column; l > 0; l--)
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
			outchar(' ');
		return nulstr;
	}

	if(strncmp(sp, "DELAY:", 6) == 0) {	// PCBoard
		mswait(atoi(sp + 6) * 100);
		return nulstr;
	}

	if(strcmp(sp, "YESCHAR") == 0) {	// PCBoard
		safe_snprintf(str, maxlen, "%c", text[YNQP][0]);
		return str;
	}

	if(strcmp(sp, "NOCHAR") == 0) {		// PCBoard
		safe_snprintf(str, maxlen, "%c", text[YNQP][1]);
		return str;
	}

	if(strcmp(sp, "QUITCHAR") == 0) {
		safe_snprintf(str, maxlen, "%c", text[YNQP][2]);
		return str;
	}

527
528
529
530
531
	if(strncmp(sp, "BPS:", 4) == 0) {
		set_output_rate((enum output_rate)atoi(sp + 4));
		return nulstr;
	}

532
533
534
535
	/* NOSTOP */

	/* STOP */

536
537
	if(!strcmp(sp,"BELL") || !strcmp(sp,"BEEP"))
		return("\a");
538

539
540
541
	if(!strcmp(sp,"EVENT")) {
		if(event_time==0)
			return("<none>");
542
		return(timestr(event_time));
543
	}
544
545
546

	/* LASTCALL */

547
	if(!strncmp(sp,"NODE",4)) {
548
549
550
		i=atoi(sp+4);
		if(i && i<=cfg.sys_nodes) {
			getnodedat(i,&node,0);
551
552
			printnodedat(i,&node);
		}
553
554
		return(nulstr);
	}
555

556
	if(!strcmp(sp,"WHO")) {
557
		whos_online(true);
558
559
		return(nulstr);
	}
560
561
562

	/* User Codes */

563
564
	if(!strcmp(sp,"USER") || !strcmp(sp,"ALIAS") || !strcmp(sp,"NAME"))
		return(useron.alias);
565

566
	if(!strcmp(sp,"FIRST")) {
567
		safe_snprintf(str,maxlen,"%s",useron.alias);
568
		tp=strchr(str,' ');
569
		if(tp) *tp=0;
570
		return(str);
571
	}
572

573
	if(!strcmp(sp,"USERNUM")) {
574
		safe_snprintf(str,maxlen,"%u",useron.number);
575
576
		return(str);
	}
577

578
	if(!strcmp(sp,"PHONE") || !strcmp(sp,"HOMEPHONE")
579
		|| !strcmp(sp,"DATAPHONE") || !strcmp(sp,"DATA"))
580
		return(useron.phone);
581

582
583
	if(!strcmp(sp,"ADDR1"))
		return(useron.address);
584

585
586
	if(!strcmp(sp,"FROM"))
		return(useron.location);
587

588
	if(!strcmp(sp,"CITY")) {
589
		safe_snprintf(str,maxlen,"%s",useron.location);
590
		char* p=strchr(str,',');
591
592
		if(p) {
			*p=0;
593
594
			return(str);
		}
595
596
		return(nulstr);
	}
597

598
599
	if(!strcmp(sp,"STATE")) {
		char* p=strchr(useron.location,',');
600
601
		if(p) {
			p++;
602
			if(*p==' ')
603
				p++;
604
605
			return(p);
		}
606
607
		return(nulstr);
	}
608

609
610
	if(!strcmp(sp,"CPU"))
		return(useron.comp);
611

612
613
	if(!strcmp(sp,"HOST"))
		return(client_name);
614

615
616
	if(!strcmp(sp,"BDATE"))
		return(useron.birth);
617

618
	if(!strcmp(sp,"AGE")) {
619
		safe_snprintf(str,maxlen,"%u",getage(&cfg,useron.birth));
620
621
		return(str);
	}
rswindell's avatar
rswindell committed
622

623
	if(!strcmp(sp,"CALLS") || !strcmp(sp,"NUMTIMESON")) {
624
		safe_snprintf(str,maxlen,"%u",useron.logons);
625
626
		return(str);
	}
627

rswindell's avatar
rswindell committed
628
629
630
631
632
633
634
	if(strcmp(sp, "PWAGE") == 0) {
		time_t age = time(NULL) - useron.pwmod;
		safe_snprintf(str, maxlen, "%ld", (long)(age/(24*60*60)));
		return str;
	}

	if(strcmp(sp, "PWDATE") == 0 || strcmp(sp, "MEMO") == 0)
635
		return(unixtodstr(&cfg,useron.pwmod,str));
636

637
	if(!strcmp(sp,"SEC") || !strcmp(sp,"SECURITY")) {
638
		safe_snprintf(str,maxlen,"%u",useron.level);
639
640
		return(str);
	}
641

642
643
	if(!strcmp(sp,"SINCE"))
		return(unixtodstr(&cfg,useron.firston,str));
644

645
	if(!strcmp(sp,"TIMEON") || !strcmp(sp,"TIMEUSED")) {
646
		now=time(NULL);
647
		safe_snprintf(str,maxlen,"%lu",(ulong)(now-logontime)/60L);
648
649
		return(str);
	}
650

651
	if(!strcmp(sp,"TUSED")) {              /* Synchronet only */
652
		now=time(NULL);
653
		return(sectostr((uint)(now-logontime),str)+1);
654
	}
655

656
	if(!strcmp(sp,"TLEFT")) {              /* Synchronet only */
657
		gettimeleft();
658
		return(sectostr(timeleft,str)+1);
659
	}
660

661
662
	if(!strcmp(sp,"TPERD"))                /* Synchronet only */
		return(sectostr(cfg.level_timeperday[useron.level],str)+1);
663

664
665
	if(!strcmp(sp,"TPERC"))                /* Synchronet only */
		return(sectostr(cfg.level_timepercall[useron.level],str)+1);
666

667
	if(!strcmp(sp,"TIMELIMIT")) {
668
		safe_snprintf(str,maxlen,"%u",cfg.level_timepercall[useron.level]);
669
670
		return(str);
	}
671

672
	if(!strcmp(sp,"MINLEFT") || !strcmp(sp,"LEFT") || !strcmp(sp,"TIMELEFT")) {
673
		gettimeleft();
674
		safe_snprintf(str,maxlen,"%lu",timeleft/60);
675
676
		return(str);
	}
677

678
	if(!strcmp(sp,"LASTON"))
679
		return(timestr(useron.laston));
680

681
682
	if(!strcmp(sp,"LASTDATEON"))
		return(unixtodstr(&cfg,useron.laston,str));
683

684
	if(!strcmp(sp,"LASTTIMEON")) {
685
		memset(&tm,0,sizeof(tm));
686
		localtime32(&useron.laston,&tm);
687
688
689
690
691
692
693
694
		if(cfg.sys_misc&SM_MILITARY)
			safe_snprintf(str,maxlen,"%02d:%02d:%02d"
				,tm.tm_hour, tm.tm_min, tm.tm_sec);
		else
			safe_snprintf(str,maxlen,"%02d:%02d %s"
				,tm.tm_hour==0 ? 12
				: tm.tm_hour>12 ? tm.tm_hour-12
				: tm.tm_hour, tm.tm_min, tm.tm_hour>11 ? "pm":"am");
695
		return(str);
696
697
	}

698
	if(!strcmp(sp,"MSGLEFT") || !strcmp(sp,"MSGSLEFT")) {
699
		safe_snprintf(str,maxlen,"%u",useron.posts);
700
701
		return(str);
	}
702

703
	if(!strcmp(sp,"MSGREAD")) {
704
		safe_snprintf(str,maxlen,"%lu",posts_read);
705
706
		return(str);
	}
707

708
	if(!strcmp(sp,"FREESPACE")) {
709
		safe_snprintf(str,maxlen,"%lu",getfreediskspace(cfg.temp_dir,0));
710
711
712
713
		return(str);
	}

	if(!strcmp(sp,"FREESPACEK")) {
714
		safe_snprintf(str,maxlen,"%lu",getfreediskspace(cfg.temp_dir,1024));
715
716
		return(str);
	}
717

718
	if(!strcmp(sp,"UPBYTES")) {
719
		safe_snprintf(str,maxlen,"%lu",useron.ulb);
720
721
		return(str);
	}
722

723
	if(!strcmp(sp,"UPK")) {
724
		safe_snprintf(str,maxlen,"%lu",useron.ulb/1024L);
725
726
		return(str);
	}
727

728
	if(!strcmp(sp,"UPS") || !strcmp(sp,"UPFILES")) {
729
		safe_snprintf(str,maxlen,"%u",useron.uls);
730
731
		return(str);
	}
732

733
	if(!strcmp(sp,"DLBYTES")) {
734
		safe_snprintf(str,maxlen,"%lu",useron.dlb);
735
736
		return(str);
	}
737

738
	if(!strcmp(sp,"DOWNK")) {
739
		safe_snprintf(str,maxlen,"%lu",useron.dlb/1024L);
740
741
		return(str);
	}
742

743
	if(!strcmp(sp,"DOWNS") || !strcmp(sp,"DLFILES")) {
744
		safe_snprintf(str,maxlen,"%u",useron.dls);
745
746
		return(str);
	}
747

748
	if(!strcmp(sp,"LASTNEW"))
749
		return(unixtodstr(&cfg,(time32_t)ns_time,str));
750

751
	if(!strcmp(sp,"NEWFILETIME"))
752
		return(timestr(ns_time));
753
754
755

	/* MAXDL */

756
	if(!strcmp(sp,"MAXDK") || !strcmp(sp,"DLKLIMIT") || !strcmp(sp,"KBLIMIT")) {
757
		safe_snprintf(str,maxlen,"%lu",cfg.level_freecdtperday[useron.level]/1024L);
758
759
		return(str);
	}
760

761
	if(!strcmp(sp,"DAYBYTES")) {    /* amt of free cdts used today */
762
		safe_snprintf(str,maxlen,"%lu",cfg.level_freecdtperday[useron.level]-useron.freecdt);
763
764
		return(str);
	}
765

766
	if(!strcmp(sp,"BYTELIMIT")) {
767
		safe_snprintf(str,maxlen,"%ld", (long)cfg.level_freecdtperday[useron.level]);
768
769
		return(str);
	}
770

771
	if(!strcmp(sp,"KBLEFT")) {
772
		safe_snprintf(str,maxlen,"%lu",(useron.cdt+useron.freecdt)/1024L);
773
774
		return(str);
	}
775

776
	if(!strcmp(sp,"BYTESLEFT")) {
777
		safe_snprintf(str,maxlen,"%lu",useron.cdt+useron.freecdt);
778
779
		return(str);
	}
780

781
	if(!strcmp(sp,"CONF")) {
782
		safe_snprintf(str,maxlen,"%s %s"
783
784
			,usrgrps ? cfg.grp[usrgrp[curgrp]]->sname :nulstr
			,usrgrps ? cfg.sub[usrsub[curgrp][cursub[curgrp]]]->sname : nulstr);
785
786
		return(str);
	}
787

788
	if(!strcmp(sp,"CONFNUM")) {
789
		safe_snprintf(str,maxlen,"%u %u",curgrp+1,cursub[curgrp]+1);
790
791
		return(str);
	}
792

793
	if(!strcmp(sp,"NUMDIR")) {
794
		safe_snprintf(str,maxlen,"%u %u",usrlibs ? curlib+1 : 0,usrlibs ? curdir[curlib]+1 : 0);
795
796
		return(str);
	}
797

798
799
	if(!strcmp(sp,"EXDATE") || !strcmp(sp,"EXPDATE"))
		return(unixtodstr(&cfg,useron.expire,str));
800

801
	if(!strcmp(sp,"EXPDAYS")) {
802
		now=time(NULL);
803
		l=(long)(useron.expire-now);
804
805
		if(l<0)
			l=0;
806
		safe_snprintf(str,maxlen,"%lu",l/(1440L*60L));
807
808
		return(str);
	}
809

810
811
	if(!strcmp(sp,"MEMO1"))
		return(useron.note);
812

813
814
	if(!strcmp(sp,"MEMO2") || !strcmp(sp,"COMPANY"))
		return(useron.name);
815

816
817
	if(!strcmp(sp,"ZIP"))
		return(useron.zipcode);
818

819
	if(!strcmp(sp,"HANGUP")) {
820
		hangup();
821
822
		return(nulstr);
	}
823
824
825

	/* Synchronet Specific */

826
	if(!strncmp(sp,"SETSTR:",7)) {
827
		strcpy(main_csi.str,sp+7);
828
829
		return(nulstr);
	}
830

831
	if(!strncmp(sp,"EXEC:",5)) {
832
		exec_bin(sp+5,&main_csi);