sbbs.h 44.9 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
/* sbbs.h */

/* Synchronet class (sbbs_t) definition and exported function prototypes */

/* $Id$ */

/****************************************************************************
 * @format.tab-size 4		(Plain Text/Source Code File Header)			*
 * @format.use-tabs true	(see http://www.synchro.net/ptsc_hdr.html)		*
 *																			*
11
 * Copyright 2014 Rob Swindell - http://www.synchro.net/copyright.html		*
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
39
40
41
42
43
44
 *																			*
 * 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.	*
 ****************************************************************************/

#ifndef _SBBS_H
#define _SBBS_H

/****************************/
/* Standard library headers */
/****************************/

rswindell's avatar
rswindell committed
45
46
47
/***************/
/* OS-specific */
/***************/
48
#if defined(_WIN32)			/* Windows */
49

deuce's avatar
deuce committed
50
51
52
	#define NOCRYPT     /* Stop windows.h from loading wincrypt.h */
                    /* Is windows.h REALLY necessary?!?! */
	#define WIN32_LEAN_AND_MEAN
53
54
55
56
57
58
	#include <io.h>
	#include <share.h>
	#include <windows.h>
	#include <process.h>	/* _beginthread() prototype */
	#include <direct.h>		/* _mkdir() prototype */
	#include <mmsystem.h>	/* SND_ASYNC */
59

60
	#if defined(_DEBUG) && defined(_MSC_VER)
61
62
63
		#include <crtdbg.h> /* Windows debug macros and stuff */
	#endif

64
#if defined(__cplusplus)
65
	extern "C"
66
#endif
deuce's avatar
deuce committed
67
	HINSTANCE hK32;
68

69
#elif defined(__unix__)		/* Unix-variant */
70

71
	#include <unistd.h>		/* close */
72
73
74

#endif

75
76
77
78
79
80
#ifdef _THREAD_SUID_BROKEN
extern int	thread_suid_broken;			/* NPTL is no longer broken */
#else
#define thread_suid_broken FALSE
#endif

rswindell's avatar
rswindell committed
81
82
83
/******************/
/* ANSI C Library */
/******************/
84
85
86
87
#include <time.h>
#include <errno.h>
#include <stdio.h>
#include <ctype.h>
88
#include <fcntl.h>			/* open */
89
90
91
#include <stdarg.h>
#include <stdlib.h>
#include <string.h>
92

93
#ifndef __unix__
94
95
96
97
98

	#include <malloc.h>

#endif

99
100
#include <sys/stat.h>

deuce's avatar
deuce committed
101
102
103
104
105
106
#ifdef __unix__
	#define XP_UNIX
#else
	#define XP_PC
	#define XP_WIN
#endif
rswindell's avatar
rswindell committed
107
108
109

#if defined(JAVASCRIPT)
#include "comio.h"			/* needed for COM_HANDLE definition only */
deuce's avatar
deuce committed
110
111
112
#include <jsversion.h>
#include <jsapi.h>
#define JS_DestroyScript(cx,script)
113

114
115
#define JSSTRING_TO_RASTRING(cx, str, ret, sizeptr, lenptr) \
{ \
116
	size_t			*JSSTSlenptr=(lenptr); \
117
118
119
120
121
122
123
124
125
	size_t			JSSTSlen; \
	size_t			JSSTSpos; \
	const jschar	*JSSTSstrval; \
	char			*JSSTStmpptr; \
\
	if(JSSTSlenptr==NULL) \
		JSSTSlenptr=&JSSTSlen; \
	if((str) != NULL) { \
		if((JSSTSstrval=JS_GetStringCharsAndLength((cx), (str), JSSTSlenptr))) { \
126
127
128
			if((*(sizeptr) < (*JSSTSlenptr+1 )) || (ret)==NULL) { \
				*(sizeptr) = *JSSTSlenptr+1; \
				if((JSSTStmpptr=(char *)realloc((ret), *(sizeptr)))==NULL) { \
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
					JS_ReportError(cx, "Error reallocating %lu bytes at %s:%d", (*JSSTSlenptr)+1, getfname(__FILE__), __LINE__); \
					(ret)=NULL; \
					free(ret); \
				} \
				else { \
					(ret)=JSSTStmpptr; \
				} \
			} \
			if(ret) { \
				for(JSSTSpos=0; JSSTSpos<*JSSTSlenptr; JSSTSpos++) \
					(ret)[JSSTSpos]=(char)JSSTSstrval[JSSTSpos]; \
				(ret)[*JSSTSlenptr]=0; \
			} \
		} \
	} \
deuce's avatar
deuce committed
144
145
	else { \
		if(ret) \
146
			*(ret)=0; \
deuce's avatar
deuce committed
147
	} \
148
149
150
151
152
153
154
155
}

#define JSVALUE_TO_RASTRING(cx, val, ret, sizeptr, lenptr) \
{ \
	JSString	*JSVTSstr=JS_ValueToString((cx), (val)); \
	JSSTRING_TO_RASTRING((cx), JSVTSstr, (ret), (sizeptr), (lenptr)); \
}

156
157
#define JSSTRING_TO_MSTRING(cx, str, ret, lenptr) \
{ \
158
	size_t			*JSSTSlenptr=(lenptr); \
159
160
161
162
163
164
165
166
167
168
169
170
171
172
	size_t			JSSTSlen; \
	size_t			JSSTSpos; \
	const jschar	*JSSTSstrval; \
\
	if(JSSTSlenptr==NULL) \
		JSSTSlenptr=&JSSTSlen; \
	(ret)=NULL; \
	if((str) != NULL) { \
		if((JSSTSstrval=JS_GetStringCharsAndLength((cx), (str), JSSTSlenptr))) { \
			if(((ret)=(char *)malloc(*JSSTSlenptr+1))) { \
				for(JSSTSpos=0; JSSTSpos<*JSSTSlenptr; JSSTSpos++) \
					(ret)[JSSTSpos]=(char)JSSTSstrval[JSSTSpos]; \
				(ret)[*JSSTSlenptr]=0; \
			} \
173
			else JS_ReportError((cx), "Error allocating %lu bytes at %s:%d", (*JSSTSlenptr)+1, getfname(__FILE__), __LINE__); \
174
175
176
177
178
179
180
181
182
183
		} \
	} \
}

#define JSVALUE_TO_MSTRING(cx, val, ret, lenptr) \
{ \
	JSString	*JSVTSstr=JS_ValueToString((cx), (val)); \
	JSSTRING_TO_MSTRING((cx), JSVTSstr, (ret), lenptr); \
}

184
185
#define JSSTRING_TO_STRBUF(cx, str, ret, bufsize, lenptr) \
{ \
186
	size_t			*JSSTSlenptr=(lenptr); \
187
188
189
190
191
192
	size_t			JSSTSlen; \
	size_t			JSSTSpos; \
	const jschar	*JSSTSstrval; \
\
	if(JSSTSlenptr==NULL) \
		JSSTSlenptr=&JSSTSlen; \
193
	if((bufsize) < 1 || (str)==NULL) \
194
195
196
		*JSSTSlenptr = 0; \
	else { \
		if((JSSTSstrval=JS_GetStringCharsAndLength((cx), (str), JSSTSlenptr))) { \
197
198
			if(*JSSTSlenptr >= (bufsize)) \
				*JSSTSlenptr = (bufsize)-1; \
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
			for(JSSTSpos=0; JSSTSpos<*JSSTSlenptr; JSSTSpos++) \
				(ret)[JSSTSpos]=(char)JSSTSstrval[JSSTSpos]; \
		} \
		else \
			*JSSTSlenptr=0; \
	} \
	(ret)[*JSSTSlenptr]=0; \
}

#define JSVALUE_TO_STRBUF(cx, val, ret, bufsize, lenptr) \
{ \
	JSString	*JSVTSstr=JS_ValueToString((cx), (val)); \
	JSSTRING_TO_STRBUF((cx), JSVTSstr, (ret), (bufsize), lenptr); \
}

#define HANDLE_PENDING(cx) \
	if(JS_IsExceptionPending(cx)) \
		return JS_FALSE;

218
219
#define JSSTRING_TO_ASTRING(cx, str, ret, maxsize, lenptr) \
{ \
220
	size_t			*JSSTSlenptr=(lenptr); \
221
222
223
224
225
226
227
228
229
	size_t			JSSTSlen; \
	size_t			JSSTSpos; \
	const jschar	*JSSTSstrval; \
\
	if(JSSTSlenptr==NULL) \
		JSSTSlenptr=&JSSTSlen; \
	(ret)=NULL; \
	if((str) != NULL) { \
		if((JSSTSstrval=JS_GetStringCharsAndLength((cx), (str), JSSTSlenptr))) { \
230
231
232
			if(*JSSTSlenptr >= (maxsize)) { \
				*JSSTSlenptr = (maxsize)-1; \
			} \
233
			if(((ret)=(char *)alloca(*JSSTSlenptr+1))) { \
234
				for(JSSTSpos=0; JSSTSpos<*JSSTSlenptr; JSSTSpos++) { \
235
					(ret)[JSSTSpos]=(char)JSSTSstrval[JSSTSpos]; \
236
				} \
237
238
				(ret)[*JSSTSlenptr]=0; \
			} \
239
			else JS_ReportError((cx), "Error allocating %lu bytes on stack at %s:%d", (*JSSTSlenptr)+1, getfname(__FILE__), __LINE__); \
240
241
242
243
244
245
246
247
248
249
		} \
	} \
}

#define JSVALUE_TO_ASTRING(cx, val, ret, maxsize, lenptr) \
{ \
	JSString	*JSVTSstr=JS_ValueToString((cx), (val)); \
	JSSTRING_TO_ASTRING((cx), JSVTSstr, (ret), (maxsize), (lenptr)); \
}

rswindell's avatar
rswindell committed
250
251
#endif

252
253
254
255
#ifdef USE_CRYPTLIB
#include <cryptlib.h>
#endif

256
/* xpdev */
257
258
259
#ifndef LINK_LIST_THREADSAFE
 #define LINK_LIST_THREADSAFE
#endif
260
#include "genwrap.h"
rswindell's avatar
rswindell committed
261
#include "semfile.h"
262
#include "netwrap.h"
263
264
#include "dirwrap.h"
#include "filewrap.h"
265
#include "datewrap.h"
266
#include "sockwrap.h"
267
#include "eventwrap.h"
268
#include "link_list.h"
269
#include "msg_queue.h"
270
#include "xpdatetime.h"
271

272
273
274
275
276
277
278
279
/***********************/
/* Synchronet-specific */
/***********************/
#include "startup.h"
#ifdef __cplusplus
	#include "threadwrap.h"	/* pthread_mutex_t */
#endif

280
281
282
283
284
285
286
287
288
#include "smblib.h"
#include "ars_defs.h"
#include "scfgdefs.h"
#include "scfglib.h"
#include "userdat.h"
#include "riodefs.h"
#include "cmdshell.h"
#include "ringbuf.h"    /* RingBuf definition */
#include "client.h"		/* client_t definition */
289
290
#include "crc16.h"
#include "crc32.h"
291
#include "telnet.h"
292
#include "nopen.h"
293
#include "text.h"
294

295
/* Synchronet Node Instance class definition */
296
297
298
299
300
301
#ifdef __cplusplus
class sbbs_t
{

public:

302
	sbbs_t(ushort node_num, SOCKADDR_IN addr, const char* host_name, SOCKET
303
		,scfg_t*, char* text[], client_t* client_info);
304
305
	~sbbs_t();

306
307
	bbs_startup_t*	startup;

308
	bool	init(void);
309
	BOOL	terminated;
310
311
312
313

	client_t client;
	SOCKET	client_socket;
	SOCKET	client_socket_dup;
314
	SOCKADDR_IN	client_addr;
315
	char	client_name[128];
rswindell's avatar
rswindell committed
316
	char	client_ident[128];
317
	DWORD	local_addr;
318
319
320
#ifdef USE_CRYPTLIB
	CRYPT_SESSION	ssh_session;
	bool	ssh_mode;
321
	SOCKET	passthru_socket;
322
323
    bool	passthru_output_thread_running;
    bool	passthru_input_thread_running;
324
#endif
325
326
327
328
329
330
331
332
333
334
335

	scfg_t	cfg;

	int		outchar_esc;		   // track ANSI escape seq output

	int 	rioctl(ushort action); // remote i/o control
	bool	rio_abortable;

    RingBuf	inbuf;
    RingBuf	outbuf;
	HANDLE	input_thread;
336
	pthread_mutex_t	input_thread_mutex;
337
	bool	input_thread_mutex_locked;	// by someone other than the input_thread
338
	pthread_mutex_t	ssh_mutex;
339
340

	int 	outcom(uchar ch); 	   // send character
341
	int 	incom(unsigned long timeout=0);		   // receive character
342

343
	void	spymsg(const char *msg);		// send message to active spies
344

345
	int		putcom(const char *str, size_t len=0);  // Send string
346
347
	void	hangup(void);		   // Hangup modem

348
349
	uchar	telnet_local_option[0x100];
	uchar	telnet_remote_option[0x100];
rswindell's avatar
rswindell committed
350
	void	send_telnet_cmd(uchar cmd, uchar opt);
351
	bool	request_telnet_opt(uchar cmd, uchar opt, unsigned waitforack=0);
352

353
    uchar	telnet_cmd[64];
rswindell's avatar
rswindell committed
354
    uint	telnet_cmdlen;
355
356
	ulong	telnet_mode;
	uchar	telnet_last_rxch;
357
	char	telnet_location[128];
358
	char	terminal[TELNET_TERM_MAXLEN+1];
359
	xpevent_t	telnet_ack_event;
360

361
	time_t	event_time;				// Time of next exclusive event
362
	char*	event_code;				// Internal code of next exclusive event
363
364
	bool	event_thread_running;
    bool	output_thread_running;
365
    bool	input_thread_running;
366

367
368
#ifdef JAVASCRIPT

369
370
371
372
373
374
375
376
	JSRuntime*		js_runtime;
	JSContext*		js_cx;
	JSObject*		js_glob;
	js_callback_t	js_callback;
	long			js_execfile(const char *fname, const char* startup_dir, JSObject* scope=NULL);
	bool			js_init(ulong* stack_frame);
	void			js_cleanup(const char* node);
	void			js_create_user_objects(void);
377
378
379

#endif

380
381
382
383
384
385
	char 	menu_dir[128];	/* Over-ride default menu dir */
	char 	menu_file[128]; /* Over-ride menu file */

	user_t	useron; 		/* User currently online */
	node_t	thisnode;		/* Node information */
	smb_t	smb;			/* Currently open message base */
386
	char	rlogin_name[LEN_ALIAS+1];
387
	char	rlogin_pass[LEN_PASS+1];
388
	char	rlogin_term[TELNET_TERM_MAXLEN+1];	/* RLogin passed terminal type/speed (e.g. "xterm/57600") */
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419

	uint	temp_dirnum;

	FILE	*nodefile_fp,
			*node_ext_fp,
			*logfile_fp;

	int 	nodefile;		/* File handle for node.dab */
	int		node_ext;		/* File handle for node.exb */
	int 	inputfile;		/* File handle to use for input */

							/* Batch download queue */
	char 	**batdn_name;	/* Filenames */
	ushort	*batdn_alt; 	/* Alternate path */
	uint 	*batdn_dir, 	/* Directory for each file */
			 batdn_total;	/* Total files */
	long 	*batdn_offset;	/* Offset for data */
	ulong	*batdn_size;	/* Size of file in bytes */
	ulong	*batdn_cdt; 	/* Credit value of file */

							/* Batch upload queue */
	char 	**batup_desc,	/* Description for each file */
			**batup_name;	/* Filenames */
	long	*batup_misc;	/* Miscellaneous bits */
	ushort	*batup_alt; 	/* Alternate path */
	uint 	*batup_dir, 	/* Directory for each file */
			batup_total;	/* Total files */

	/*********************************/
	/* Color Configuration Variables */
	/*********************************/
420
421
	char 	*text[TOTAL_TEXT];			/* Text from ctrl\text.dat */
	char 	*text_sav[TOTAL_TEXT];		/* Text from ctrl\text.dat */
422

423
	char 	dszlog[127];	/* DSZLOG enviornment variable */
424
    int     keybuftop,keybufbot;    /* Keyboard input buffer pointers (for ungetkey) */
425
	char    keybuf[KEY_BUFSIZE];    /* Keyboard input buffer */
426
427

	ushort	node_connection;
428
	char	connection[LEN_MODEM+1];	/* Connection Description */
429
430
431
432
433
434
	ulong	cur_rate;		/* Current Connection (DCE) Rate */
	ulong	cur_cps;		/* Current Average Transfer CPS */
	ulong	dte_rate;		/* Current COM Port (DTE) Rate */
	time_t 	timeout;		/* User inactivity timeout reference */
	ulong 	timeleft_warn;	/* low timeleft warning flag */
	uchar 	curatr; 		/* Current Text Attributes Always */
435
436
	uchar	attr_stack[64];	/* Saved attributes (stack) */
	int 	attr_sp;		/* Attribute stack pointer */
437
438
	long 	lncntr; 		/* Line Counter - for PAUSE */
	long 	tos;			/* Top of Screen */
439
440
	long 	rows;			/* Current number of Rows for User */
	long	cols;			/* Current number of Columns for User */
441
	long	column;			/* Current column counter (for line counter) */
442
443
444
	long 	autoterm;		/* Autodetected terminal type */
	char 	slbuf[SAVE_LINES][LINE_BUFSIZE+1]; /* Saved for redisplay */
	char 	slatr[SAVE_LINES];	/* Starting attribute of each line */
445
446
	char 	slcuratr[SAVE_LINES];	/* Ending attribute of each line */
	int 	slcnt;			/* Number of lines currently saved */
447
448
449
450
451
	char 	lbuf[LINE_BUFSIZE+1];/* Temp storage for each line output */
	int		lbuflen;		/* Number of characters in line buffer */
	char 	latr;			/* Starting attribute of line buffer */
	ulong	console;		/* Defines current Console settings */
	char 	wordwrap[81];	/* Word wrap buffer */
452
	time_t	now,			/* Used to store current time in Unix format */
453
			answertime, 	/* Time call was answered */
454
455
			logontime,		/* Time user logged on */
			starttime,		/* Time stamp to use for time left calcs */
456
			ns_time,		/* File new-scan time */
457
458
459
460
			last_ns_time;	/* Most recent new-file-scan this call */
	uchar 	action;			/* Current action of user */
	long 	online; 		/* Remote/Local or not online */
	long 	sys_status; 	/* System Status */
461
	subscan_t	*subscan;	/* User sub configuration/scan info */
462
463
464
465
466
467
468
469

	ulong	logon_ulb,		/* Upload Bytes This Call */
			logon_dlb,		/* Download Bytes This Call */
			logon_uls,		/* Uploads This Call */
			logon_dls,		/* Downloads This Call */
			logon_posts,	/* Posts This Call */
			logon_emails,	/* Emails This Call */
			logon_fbacks;	/* Feedbacks This Call */
470
	uchar	logon_ml;		/* ML of the user upon logon */
471
472
473
474
475
476
477

	uint 	main_cmds;		/* Number of Main Commands this call */
	uint 	xfer_cmds;		/* Number of Xfer Commands this call */
	ulong	posts_read; 	/* Number of Posts read this call */
	char 	temp_uler[31];  /* User who uploaded the files to temp dir */
	char 	temp_file[41];	/* Origin of extracted temp files */
	long 	temp_cdt;		/* Credit value of file that was extracted */
478
	bool 	autohang;		/* Used for auto-hangup after transfer */
479
480
481
	size_t 	logcol; 		/* Current column of log file */
	uint 	criterrs; 		/* Critical error counter */

482
483
484
485
486
487
	uint 	curgrp; 		/* Current group */
	uint	*cursub;		/* Current sub-board for each group */
	uint	curlib; 		/* Current library */
	uint	*curdir;		/* Current directory for each library */
	uint 	*usrgrp;		/* Real group numbers */
	uint	usrgrps;		/* Number groups this user has access to */
488
	uint	usrgrp_total;	/* Total number of groups */
489
490
	uint 	*usrlib;		/* Real library numbers */
	uint	usrlibs;		/* Number of libs this user can access */
491
	uint	usrlib_total;	/* Total number of libraries */
492
493
494
495
	uint 	**usrsub;		/* Real sub numbers */
	uint	*usrsubs;		/* Num of subs with access for each grp */
	uint 	**usrdir;		/* Real dir numbers */
	uint	*usrdirs;		/* Num of dirs with access for each lib */
496
497
498
	uint	cursubnum;		/* For ARS */
	uint	curdirnum;		/* For ARS */
	ulong 	timeleft;		/* Number of seconds user has left online */
499

500
501
502
503
504
505
506
507
508
	char 	*comspec;		/* Pointer to environment variable COMSPEC */
	ushort	altul;			/* Upload to alternate path flag */
	char 	cid[LEN_CID+1]; /* Caller ID (IP Address) of current caller */
	char 	*noaccess_str;	/* Why access was denied via ARS */
	long 	noaccess_val;	/* Value of parameter not met in ARS */
	int		errorlevel; 	/* Error level of external program */

	csi_t	main_csi;		/* Main Command Shell Image */

509
510
	smbmsg_t*	current_msg;	/* For message header @-codes */

511
512
513
			/* Global command shell variables */
	uint	global_str_vars;
	char **	global_str_var;
deuce's avatar
deuce committed
514
	int32_t *	global_str_var_name;
515
	uint	global_int_vars;
deuce's avatar
deuce committed
516
	int32_t *	global_int_var;
deuce's avatar
deuce committed
517
	int32_t *	global_int_var_name;
518
519
	char *	sysvar_p[MAX_SYSVARS];
	uint	sysvar_pi;
deuce's avatar
deuce committed
520
	int32_t	sysvar_l[MAX_SYSVARS];
521
522
523
	uint	sysvar_li;

    /* ansi_term.cpp */
524
525
	const char*	ansi(int atr);			/* Returns ansi escape sequence for atr */
	char*	ansi(int atr, int curatr, char* str);
526
527
528
529
    bool	ansi_gotoxy(int x, int y);
	bool	ansi_getxy(int* x, int* y);
	bool	ansi_save(void);
	bool	ansi_restore(void);
530
531
532
533
534
535
	void	ansi_getlines(void);

			/* Command Shell Methods */
	int		exec(csi_t *csi);
	int		exec_function(csi_t *csi);
	int		exec_misc(csi_t *csi, char *path);
rswindell's avatar
rswindell committed
536
537
538
	int		exec_net(csi_t *csi);
	int		exec_msg(csi_t *csi);
	int		exec_file(csi_t *csi);
539
	long	exec_bin(const char *mod, csi_t *csi, const char* startup_dir=NULL);
540
541
	void	clearvars(csi_t *bin);
	void	freevars(csi_t *bin);
deuce's avatar
deuce committed
542
	char**	getstrvar(csi_t *bin, int32_t name);
deuce's avatar
deuce committed
543
	int32_t*	getintvar(csi_t *bin, int32_t name);
544
545
	char*	copystrvar(csi_t *csi, char *p, char *str);
	void	skipto(csi_t *csi, uchar inst);
546
	bool	ftp_cmd(csi_t* csi, SOCKET ctrl_sock, const char* cmdsrc, char* rsp);
rswindell's avatar
rswindell committed
547
	bool	ftp_put(csi_t* csi, SOCKET ctrl_sock, char* src, char* dest);
548
549
550
	bool	ftp_get(csi_t* csi, SOCKET ctrl_sock, char* src, char* dest, bool dir=false);
	SOCKET	ftp_data_sock(csi_t* csi, SOCKET ctrl_sock, SOCKADDR_IN*);

551
552
553
	bool	select_shell(void);
	bool	select_editor(void);

554
555
556
	void	sys_info(void);
	void	user_info(void);
	void	xfer_policy(void);
557
558

	void	xfer_prot_menu(enum XFER_TYPE);
559
560
561
562
	void	node_stats(uint node_num);
	void	sys_stats(void);
	void	logonlist(void);
	bool	spy(uint node_num);
563
564
565
566
567
568

	void	reset_logon_vars(void);

	uint	finduser(char *str);

	int 	sub_op(uint subnum);
deuce's avatar
deuce committed
569
	ulong	getlastmsg(uint subnum, uint32_t *ptr, time_t *t);
570
571
572
573
574
575
576
577
578
579
	time_t	getmsgtime(uint subnum, ulong ptr);
	ulong	getmsgnum(uint subnum, time_t t);

	int		dir_op(uint dirnum);
	int		getuserxfers(int fromuser, int destuser, char *fname);

	void	getmsgptrs(void);
	void	putmsgptrs(void);
	void	getusrsubs(void);
	void	getusrdirs(void);
580
581
582
	uint	getusrsub(uint subnum);
	uint	getusrgrp(uint subnum);

583
	uint	userdatdupe(uint usernumber, uint offset, uint datlen, char *dat
584
				,bool del=false, bool next=false);
585
	ulong	gettimeleft(bool handle_out_of_time=true);
586
587
588
	bool	gettimeleft_inside;

	/* str.cpp */
589
	char*	timestr(time_t intime);
590
    char	timestr_output[60];
591
	void	userlist(long mode);
592
	size_t	gettmplt(char *outstr, const char *tmplt, long mode);
593
	void	sif(char *fname, char *answers, long len);	/* Synchronet Interface File */
594
595
596
597
	void	sof(char *fname, char *answers, long len);
	void	create_sif_dat(char *siffile, char *datfile);
	void	read_sif_dat(char *siffile, char *datfile);
	void	printnodedat(uint number, node_t* node);
deuce's avatar
deuce committed
598
	bool	inputnstime32(time32_t *dt);
599
	bool	inputnstime(time_t *dt);
600
	bool	chkpass(char *pass, user_t* user, bool unique);
601
	char *	cmdstr(const char *instr, const char *fpath, const char *fspec, char *outstr);
602
603
604
605
	char	cmdstr_output[512];

	void	subinfo(uint subnum);
	void	dirinfo(uint dirnum);
606
	bool	trashcan(const char *insearch, const char *name);
607
608
	void	time_bank(void);
	void	change_user(void);
609
610
611

	/* writemsg.cpp */
	void	automsg(void);
612
	bool	writemsg(const char *str, const char *top, char *title, long mode, uint subnum
613
				,const char *dest, char** editor=NULL);
614
	char*	quotes_fname(int xedit, char* buf, size_t len);
615
	char*	msg_tmp_fname(int xedit, char* fname, size_t len);
616
	char	putmsg(const char *str, long mode);
617
	bool	msgabort(void);
618
	bool	email(int usernumber, const char *top, const char *title, long mode);
619
620
	void	forwardmail(smbmsg_t* msg, int usernum);
	void	removeline(char *str, char *str2, char num, char skip);
621
	ulong	msgeditor(char *buf, const char *top, char *title);
622
	bool	editfile(char *path, bool msg=false);
623
624
625
626
627
628
629
630
	int		loadmsg(smbmsg_t *msg, ulong number);
	ushort	chmsgattr(ushort attr);
	void	show_msgattr(ushort attr);
	void	show_msghdr(smbmsg_t* msg);
	void	show_msg(smbmsg_t* msg, long mode);
	void	msgtotxt(smbmsg_t* msg, char *str, int header, int tails);
	void	quotemsg(smbmsg_t* msg, int tails);
	void	editmsg(smbmsg_t* msg, uint subnum);
631
	void	editor_inf(int xeditnum, const char *dest, const char *title, long mode
632
633
634
				,uint subnum);
	void	copyfattach(uint to, uint from, char *title);
	bool	movemsg(smbmsg_t* msg, uint subnum);
635
636
	int		process_edited_text(char* buf, FILE* stream, long mode, unsigned* lines, unsigned maxlines);
	int		process_edited_file(const char* src, const char* dest, long mode, unsigned* lines, unsigned maxlines);
637

638
639
640
	/* postmsg.cpp */
	bool	postmsg(uint subnum, smbmsg_t* msg, long wm_mode);

641
642
643
	/* mail.cpp */
	int		delmail(uint usernumber,int which);
	void	telluser(smbmsg_t* msg);
644
	void	delallmail(uint usernumber, int which, bool permanent=true);
645
646

	/* getmsg.cpp */
647
	post_t* loadposts(uint32_t *posts, uint subnum, ulong ptr, long mode, ulong *unvalidated_num);
648
649
650
651
652

	/* readmail.cpp */
	void	readmail(uint usernumber, int sent);

	/* bulkmail.cpp */
653
654
	bool	bulkmail(uchar *ar);
	int		bulkmailhdr(smb_t*, smbmsg_t*, uint usernum);
655
656

	/* con_out.cpp */
657
658
659
660
	int		bputs(const char *str);					/* BBS puts function */
	int		rputs(const char *str, size_t len=0);	/* BBS raw puts function */
	int		bprintf(const char *fmt, ...);			/* BBS printf function */
	int		rprintf(const char *fmt, ...);			/* BBS raw printf function */
661
	void	backspace(void);				/* Output a destructive backspace via outchar */
662
	void	outchar(char ch);				/* Output a char - check echo and emu.  */
663
	void	center(char *str);
664
665
666
667
668
669
670
	void	clearline(void);
	void	cleartoeol(void);
	void	cursor_home(void);
	void	cursor_up(int count=1);
	void	cursor_down(int count=1);
	void	cursor_left(int count=1);
	void	cursor_right(int count=1);
671
	long	term_supports(long cmp_flags=0);
672
673

	/* getstr.cpp */
674
	size_t	getstr_offset;
675
	size_t	getstr(char *str, size_t length, long mode);
676
	long	getnum(ulong max, ulong dflt=0);
677
	void	insert_indicator(void);
678
679

	/* getkey.cpp */
680
	char	getkey(long mode); 		/* Waits for a key hit local or remote  */
681
	long	getkeys(const char *str, ulong max);
682
	void	ungetkey(char ch);		/* Places 'ch' into the input buffer    */
683
	char	question[MAX_TEXTDAT_ITEM_LEN+1];
684
685
	bool	yesno(const char *str);
	bool	noyes(const char *str);
686
	void	pause(void);
687
688
	const char *	mnestr;
	void	mnemonics(const char *str);
689

rswindell's avatar
rswindell committed
690
	/* inkey.cpp */
691
	char	inkey(long mode, unsigned long timeout=0);
rswindell's avatar
rswindell committed
692
693
	char	handle_ctrlkey(char ch, long mode=0);

694
695
696
	/* prntfile.cpp */
	void	printfile(char *str, long mode);
	void	printtail(char *str, int lines, long mode);
697
	void	menu(const char *code);
698

699
	int		uselect(int add, uint n, const char *title, const char *item, const uchar *ar);
700
701
702
703
704
705
706
	uint	uselect_total, uselect_num[500];

	void	redrwstr(char *strin, int i, int l, long mode);
	void	attr(int atr);				/* Change local and remote text attributes */
	void	ctrl_a(char x);			/* Peforms the Ctrl-Ax attribute changes */

	/* atcodes.cpp */
707
708
	int		show_atcode(const char *code);
	const char*	atcode(char* sp, char* str, size_t maxlen);
709
710

	/* getnode.cpp */
711
712
	int		getsmsg(int usernumber);
	int		getnmsg(void);
713
	int		whos_online(bool listself);/* Lists active nodes, returns active nodes */
714
	void	nodelist(void);
715
716
717
718
719
720
721
722
723
	int		getnodeext(uint number, char * str);
	int		getnodedat(uint number, node_t * node, bool lock);
	void	nodesync(void);
	user_t	nodesync_user;
	bool	nodesync_inside;

	/* putnode.cpp */
	int		putnodedat(uint number, node_t * node);
	int		putnodeext(uint number, char * str);
724

725
	/* login.ccp */
726
	int		login(char *str, char *pw);
727
728
729
730
731
732
	void	badlogin(char* user, char* passwd);

	/* answer.cpp */
	bool	answer();

	/* logon.ccp */
733
	bool	logon(void);
734
735

	/* logout.cpp */
736
737
738
	void	logout(void);
	void	backout(void);

739
740
741
	/* newuser.cpp */
	BOOL	newuser(void);					/* Get new user							*/

742
	/* text_sec.cpp */
743
	int		text_sec(void);						/* Text sections */
744
745

	/* readmsgs.cpp */
746
747
	int		scanposts(uint subnum, long mode, const char* find);	/* Scan sub-board */
	long	listsub(uint subnum, long mode, long start, const char* search);
748
	long	listmsgs(uint subnum, long mode, post_t* post, long start, long posts);
749
	long	searchposts(uint subnum, post_t* post, long start, long msgs, const char* find);
750
	long	showposts_toyou(post_t* post, ulong start, long posts);
751
752
753
754
	void	msghdr(smbmsg_t* msg);

	/* chat.cpp */
	void	chatsection(void);
rswindell's avatar
rswindell committed
755
	void	multinodechat(int channel=1);
756
757
	void	nodepage(void);
	void	nodemsg(void);
758
759
	uint	nodemsg_inside;
	uint	hotkey_inside;
760
761
	uchar	lastnodemsg;	/* Number of node last message was sent to */
	char	lastnodemsguser[LEN_ALIAS+1];
762
	void	guruchat(char* line, char* guru, int gurunum, char* last_answer);
763
764
	bool	guruexp(char **ptrptr, char *line);
	void	localguru(char *guru, int gurunum);
rswindell's avatar
rswindell committed
765
766
	bool	sysop_page(void);
	bool	guru_page(void);
767
768
769
770
771
772
773
774
	void	privchat(bool local=false);
	bool	chan_access(uint cnum);
	int		getnodetopage(int all, int telegram);

	/* main.cpp */
	void	printstatslog(uint node);
	ulong	logonstats(void);
	void	logoffstats(void);
775
	int		nopen(char *str, int access);
776
	int		mv(char *src, char *dest, char copy); /* fast file move/copy function */
777
	bool	chksyspass(void);
rswindell's avatar
rswindell committed
778
779
	bool	chk_ar(const uchar * str, user_t* user, client_t* client); /* checks access requirements */
	bool	ar_exp(const uchar ** ptrptr, user_t*, client_t*);
780
	void	daily_maint(void);
781
782
783
784

	/* upload.cpp */
	bool	uploadfile(file_t* f);
	char	sbbsfilename[128],sbbsfiledesc[128]; /* env vars */
785
	bool	upload(uint dirnum);
786
787
788
789
790
791
    char	upload_lastdesc[LEN_FDESC+1];
	bool	bulkupload(uint dirnum);

	/* download.cpp */
	void	downloadfile(file_t* f);
	void	notdownloaded(ulong size, time_t start, time_t end);
792
	int		protocol(prot_t* prot, enum XFER_TYPE, char *fpath, char *fspec, bool cd);
793
	const char*	protcmdline(prot_t* prot, enum XFER_TYPE type);
794
795
	void	seqwait(uint devnum);
	void	autohangup(void);
796
797
	bool	checkdszlog(file_t*);
	bool	checkprotresult(prot_t*, int error, file_t*);
798
799
	bool	sendfile(char* fname, char prot=0);
	bool	recvfile(char* fname, char prot=0);
800
801
802
803
804
805

	/* file.cpp */
	void	fileinfo(file_t* f);
	void	openfile(file_t* f);
	void	closefile(file_t* f);
	bool	removefcdt(file_t* f);
806
	bool	removefile(file_t* f);
807
808
809
810
811
812
	bool	movefile(file_t* f, int newdir);
	char *	getfilespec(char *str);
	bool	checkfname(char *fname);
	bool	addtobatdl(file_t* f);

	/* listfile.cpp */
813
814
815
	bool	listfile(const char *fname, const char *buf, uint dirnum
				,const char *search, const char letter, ulong datoffset);
	int		listfiles(uint dirnum, const char *filespec, int tofile, long mode);
816
	int		listfileinfo(uint dirnum, char *filespec, long mode);
deuce's avatar
deuce committed
817
	void	listfiletofile(char *fname, char *buf, uint dirnum, int file);
818
819
820
821
	int		batchflagprompt(uint dirnum, file_t bf[], uint total, long totalfiles);

	/* bat_xfer.cpp */
	void	batchmenu(void);
822
	void	batch_create_list(void);
823
824
	void	batch_add_list(char *list);
	bool	create_batchup_lst(void);
825
	bool	create_batchdn_lst(bool native);
826
827
828
	bool	create_bimodem_pth(void);
	void	batch_upload(void);
	void	batch_download(int xfrprot);
829
	BOOL	start_batch_download(void);
830
831
832
833
834

	/* tmp_xfer.cpp */
	void	temp_xfer(void);
	void	extract(uint dirnum);
	char *	temp_cmd(void);					/* Returns temp file command line */
835
	ulong	create_filelist(const char *name, long mode);
836
837
838
839
840
841
842
843
844
845

	/* viewfile.cpp */
	int		viewfile(file_t* f, int ext);
	void	viewfiles(uint dirnum, char *fspec);
	void	viewfilecontents(file_t* f);

	/* sortdir.cpp */
	void	resort(uint dirnum);

	/* xtrn.cpp */
846
	int		external(const char* cmdline, long mode, const char* startup_dir=NULL);
847
848
849
850
851

	/* xtrn_sec.cpp */
	int		xtrn_sec(void);					/* The external program section  */
	void	xtrndat(char* name, char* dropdir, uchar type, ulong tleft
				,ulong misc);
852
853
	bool	exec_xtrn(uint xtrnnum);			/* Executes online external program */
	bool	user_event(user_event_t);			/* Executes user event(s) */
854
855
856
	char	xtrn_access(uint xnum);			/* Does useron have access to xtrn? */
	void	moduserdat(uint xtrnnum);

rswindell's avatar
rswindell committed
857
	/* logfile.cpp */
858
	void	logentry(const char *code,const char *entry);
859
	void	log(char *str);				/* Writes 'str' to node log */
860
	void	logch(char ch, bool comma);	/* Writes 'ch' to node log */
861
862
	void	logline(const char *code,const char *str); /* Writes 'str' on it's own line in log (LOG_INFO level) */
	void	logline(int level, const char *code,const char *str);
863
	void	logofflist(void);              /* List of users logon activity */
864
	bool	syslog(const char* code, const char *entry);
865
	bool	errormsg_inside;
866
	void	errormsg(int line, const char *file, const char* action, const char *object
867
				,ulong access, const char *extinfo=NULL);
868
	BOOL	hacklog(char* prot, char* text);
869

870
871
	/* qwk.cpp */
	bool	qwklogon;
872
	ulong	qwkmail_last;
873
874
	void	qwk_sec(void);
	int		qwk_route(char *inaddr, char *fulladdr);
875
876
877
878
879
880
	uint	total_qwknodes;
	struct qwknode {
		char	id[LEN_QWKID+1];
		char	path[MAX_PATH+1];
		time_t	time;
	}* qwknode;
881
882
883
884
	void	update_qwkroute(char *via);
	void	qwk_success(ulong msgcnt, char bi, char prepack);
	void	qwksetptr(uint subnum, char *buf, int reset);
	void	qwkcfgline(char *buf,uint subnum);
885
	int		set_qwk_flag(ulong flag);
886
887
888
889
890

	/* pack_qwk.cpp */
	bool	pack_qwk(char *packet, ulong *msgcnt, bool prepack);

	/* un_qwk.cpp */
891
	bool	unpack_qwk(char *packet,uint hubnum);
892
893
894
895
896
897

	/* pack_rep.cpp */
	bool	pack_rep(uint hubnum);

	/* un_rep.cpp */
	bool	unpack_rep(char* repfile=NULL);
898
	uint	resolve_qwkconf(uint n);
899
900

	/* msgtoqwk.cpp */
901
	ulong	msgtoqwk(smbmsg_t* msg, FILE *qwk_fp, long mode, uint subnum, int conf, FILE* hdrs_dat);
902
903

	/* qwktomsg.cpp */
904
	void	qwk_new_msg(smbmsg_t* msg, char* hdrblk, long offset, str_list_t headers, bool parse_sender_hfields);
rswindell's avatar
rswindell committed
905
906
	bool	qwk_import_msg(FILE *qwk_fp, char *hdrblk, ulong blocks, char fromhub, uint subnum
				,uint touser, smbmsg_t* msg);
907
908

	/* fido.cpp */
909
	bool	netmail(const char *into, const char *subj, long mode);
910
911
912
	void	qwktonetmail(FILE *rep, char *block, char *into, uchar fromhub);
	bool	lookup_netuser(char *into);

913
914
	bool	inetmail(const char *into, const char *subj, long mode);
	bool	qnetmail(const char *into, const char *subj, long mode);
915
916

	/* useredit.cpp */
917
	void	useredit(int usernumber);
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
	int		searchup(char *search,int usernum);
	int		searchdn(char *search,int usernum);
	void	maindflts(user_t* user);
	void	purgeuser(int usernumber);

	/* ver.cpp */
	void	ver(void);

	/* scansubs.cpp */
	void	scansubs(long mode);
	void	scanallsubs(long mode);
	void	new_scan_cfg(ulong misc);
	void	new_scan_ptr_cfg(void);

	/* scandirs.cpp */
	void	scanalldirs(long mode);
	void	scandirs(long mode);

	#define nosound()
	#define checkline()

	void	catsyslog(int crash);

	/* telgate.cpp */
942
	void	telnet_gate(char* addr, ulong mode, char* client_user_name=NULL, char* server_user_name=NULL, char* term_type=NULL);	// See TG_* for mode bits
943
944

};
945

rswindell's avatar
rswindell committed
946
#endif /* __cplusplus */
947

948
949
950
#ifdef DLLEXPORT
#undef DLLEXPORT
#endif
951
952
953
#ifdef DLLCALL
#undef DLLCALL
#endif
954
#ifdef _WIN32
deuce's avatar
deuce committed
955
956
	#ifdef __MINGW32__
		#define DLLEXPORT
957
		#define DLLCALL
deuce's avatar
deuce committed
958
959
960
961
962
963
964
965
966
967
968
	#else
		#ifdef SBBS_EXPORTS
			#define DLLEXPORT	__declspec(dllexport)
		#else
			#define DLLEXPORT	__declspec(dllimport)
		#endif
		#ifdef __BORLANDC__
			#define DLLCALL __stdcall
		#else
			#define DLLCALL
		#endif
969
970
971
	#endif
#else	/* !_WIN32 */
	#define DLLEXPORT
972
	#define DLLCALL
973
974
975
976
977
978
#endif

#ifdef __cplusplus
extern "C" {
#endif

979
980
981
982
	/* main.cpp */
	DLLEXPORT int		DLLCALL sbbs_random(int);
	DLLEXPORT void		DLLCALL sbbs_srand(void);

983
	/* getstats.c */
984
	DLLEXPORT BOOL		DLLCALL getstats(scfg_t* cfg, char node, stats_t* stats);
985
986
	DLLEXPORT ulong		DLLCALL	getposts(scfg_t* cfg, uint subnum);
	DLLEXPORT long		DLLCALL getfiles(scfg_t* cfg, uint dirnum);
987

988
	/* getmail.c */
989
	DLLEXPORT int		DLLCALL getmail(scfg_t* cfg, int usernumber, BOOL sent);
990
	DLLEXPORT mail_t *	DLLCALL loadmail(smb_t* smb, uint32_t* msgs, uint usernumber
991
										,int which, long mode);
992
	DLLEXPORT void		DLLCALL freemail(mail_t* mail);
993
	DLLEXPORT void		DLLCALL delfattach(scfg_t*, smbmsg_t*);
994

995
	/* postmsg.cpp */
996
	DLLEXPORT int		DLLCALL savemsg(scfg_t*, smb_t*, smbmsg_t*, client_t*, const char* server, char* msgbuf);
997
998
	DLLEXPORT void		DLLCALL signal_sub_sem(scfg_t*, uint subnum);
	DLLEXPORT int		DLLCALL msg_client_hfields(smbmsg_t*, client_t*);
999
	DLLEXPORT char*		DLLCALL msg_program_id(char* pid);
1000

1001
	/* filedat.c */
1002
	DLLEXPORT BOOL		DLLCALL getfileixb(scfg_t* cfg, file_t* f);
1003
	DLLEXPORT BOOL		DLLCALL putfileixb(scfg_t* cfg, file_t* f);
1004
1005
1006
1007
	DLLEXPORT BOOL		DLLCALL getfiledat(scfg_t* cfg, file_t* f);
	DLLEXPORT BOOL		DLLCALL putfiledat(scfg_t* cfg, file_t* f);
	DLLEXPORT void		DLLCALL putextdesc(scfg_t* cfg, uint dirnum, ulong datoffset, char *ext);
	DLLEXPORT void		DLLCALL getextdesc(scfg_t* cfg, uint dirnum, ulong datoffset, char *ext);
1008
	DLLEXPORT char*		DLLCALL getfilepath(scfg_t* cfg, file_t* f, char* path);
1009
1010
1011
1012

	DLLEXPORT BOOL		DLLCALL removefiledat(scfg_t* cfg, file_t* f);
	DLLEXPORT BOOL		DLLCALL addfiledat(scfg_t* cfg, file_t* f);
	DLLEXPORT BOOL		DLLCALL findfile(scfg_t* cfg, uint dirnum, char *filename);
1013
1014
	DLLEXPORT char *	DLLCALL padfname(const char *filename, char *str);
	DLLEXPORT char *	DLLCALL unpadfname(const char *filename, char *str);
1015
1016
	DLLEXPORT BOOL		DLLCALL rmuserxfers(scfg_t* cfg, int fromuser, int destuser, char *fname);

1017
	DLLEXPORT int		DLLCALL update_uldate(scfg_t* cfg, file_t* f);
1018

1019
	/* str_util.c */
1020
1021
	DLLEXPORT char *	DLLCALL remove_ctrl_a(const char* instr, char* outstr);
	DLLEXPORT char 		DLLCALL ctrl_a_to_ascii_char(char code);
1022
	DLLEXPORT char *	DLLCALL truncstr(char* str, const char* set);
1023
	DLLEXPORT char *	DLLCALL ascii_str(uchar* str);
1024
	DLLEXPORT char		DLLCALL exascii_to_ascii_char(uchar ch);
rswindell's avatar
rswindell committed
1025
	DLLEXPORT BOOL		DLLCALL findstr(const char *insearch, const char *fname);
1026
	DLLEXPORT BOOL		DLLCALL findstr_in_string(const char* insearchof, char* string);
rswindell's avatar
rswindell committed
1027
1028
1029
1030
	DLLEXPORT BOOL		DLLCALL findstr_in_list(const char* insearchof, str_list_t list);
	DLLEXPORT BOOL		DLLCALL trashcan(scfg_t* cfg, const char *insearch, const char *name);
	DLLEXPORT char *	DLLCALL trashcan_fname(scfg_t* cfg, const char *name, char* fname, size_t);
	DLLEXPORT str_list_t DLLCALL trashcan_list(scfg_t* cfg, const char* name);
1031
	DLLEXPORT char *	DLLCALL strip_exascii(const char *str, char* dest);
1032
	DLLEXPORT char *	DLLCALL strip_space(const char *str, char* dest);
1033
1034
	DLLEXPORT char *	DLLCALL prep_file_desc(const char *str, char* dest);
	DLLEXPORT char *	DLLCALL strip_ctrl(const char *str, char* dest);
1035
	DLLEXPORT char *	DLLCALL net_addr(net_t* net);
1036
1037
	DLLEXPORT BOOL		DLLCALL valid_ctrl_a_attr(char a);
	DLLEXPORT BOOL		DLLCALL valid_ctrl_a_code(char a);
1038
	DLLEXPORT size_t	DLLCALL strip_invalid_attr(char *str);
1039
	DLLEXPORT char *	DLLCALL ultoac(ulong l,char *str);
1040
	DLLEXPORT char *	DLLCALL rot13(char* str);
1041
	DLLEXPORT uint32_t	DLLCALL str_to_bits(uint32_t currval, const char *str);
1042