addfiles.c 24.8 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
/* addfiles.c */

/* Program to add files to a Synchronet file database */

/* $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 2009 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
 *																			*
 * 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"

40
#define ADDFILES_VER "3.02"
41
42
43
44
45
46
47

scfg_t scfg;

int cur_altpath=0;

long files=0,removed=0,mode=0;

48
49
char lib[LEN_GSNAME+1];

50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
#define DEL_LIST	(1L<<0)
#define NO_EXTEND	(1L<<1)
#define FILE_DATE	(1L<<2)
#define TODAYS_DATE (1L<<3)
#define FILE_ID 	(1L<<4)
#define NO_UPDATE	(1L<<5)
#define NO_NEWDATE	(1L<<6)
#define ASCII_ONLY	(1L<<7)
#define UL_STATS	(1L<<8)
#define ULDATE_ONLY (1L<<9)
#define KEEP_DESC	(1L<<10)
#define AUTO_ADD	(1L<<11)
#define SEARCH_DIR	(1L<<12)
#define SYNC_LIST	(1L<<13)
#define KEEP_SPACE	(1L<<14)

/****************************************************************************/
67
/* This is needed by load_cfg.c												*/
68
/****************************************************************************/
69
int lprintf(int level, const char *fmat, ...)
70
71
{
	va_list argptr;
72
	char sbuf[512];
73
74
75
76
77
	int chcount;

	va_start(argptr,fmat);
	chcount=vsprintf(sbuf,fmat,argptr);
	va_end(argptr);
78
79
	truncsp(sbuf);
	printf("%s\n",sbuf);
80
81
82
	return(chcount);
}

83
void prep_desc(char *str)
84
85
86
87
88
{
	char tmp[1024];
	int i,j;

	for(i=j=0;str[i] && j < sizeof(tmp)-1;i++) {
89
		if(j && str[i]==' ' && tmp[j-1]==' ' && (mode&KEEP_SPACE))
90
			tmp[j++]=str[i];
91
		else if(j && str[i]<=' ' && str[i] > 0&& tmp[j-1]==' ')
92
			continue;
93
		else if(i && !isalnum((uchar)str[i]) && str[i]==str[i-1])
94
			continue;
95
		else if(str[i]>=' ' || str[i]<0)
96
97
			tmp[j++]=str[i];
		else if(str[i]==TAB || (str[i]==CR && str[i+1]==LF))
98
			tmp[j++]=' ';
99
100
101
102
103
	}
	tmp[j]=0;
	strcpy(str,tmp);
}

104
105
106
/*****************************************************************************/
/* Returns command line generated from instr with %c replacments             */
/*****************************************************************************/
107
char *mycmdstr(char *instr, char *fpath, char *fspec, char *outstr)
108
109
110
111
{
    static char cmd[MAX_PATH+1];
    char str[MAX_PATH+1];
    int i,j,len;
rswindell's avatar
rswindell committed
112
113
114
#ifdef _WIN32
	char sfpath[MAX_PATH+1];
#endif
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154

	len=strlen(instr);
	for(i=j=0;i<len && j<128;i++) {
		if(instr[i]=='%') {
			i++;
			cmd[j]=0;
			switch(toupper(instr[i])) {
				case 'F':   /* File path */
					strcat(cmd,fpath);
					break;
				case '~':	/* DOS-compatible (8.3) filename */
#ifdef _WIN32
					SAFECOPY(sfpath,fpath);
					GetShortPathName(fpath,sfpath,sizeof(sfpath));
					strcat(cmd,sfpath);
#else
                    strcat(cmd,fpath);
#endif			
					break;
				case 'G':   /* Temp directory */
					strcat(cmd,scfg.temp_dir);
					break;
                case 'J':
                    strcat(cmd,scfg.data_dir);
                    break;
                case 'K':
                    strcat(cmd,scfg.ctrl_dir);
                    break;
				case 'N':   /* Node Directory (same as SBBSNODE environment var) */
					strcat(cmd,scfg.node_dir);
					break;
				case 'S':   /* File Spec */
					strcat(cmd,fspec);
					break;
                case 'Z':
                    strcat(cmd,scfg.text_dir);
                    break;
				case '!':   /* EXEC Directory */
					strcat(cmd,scfg.exec_dir);
					break;
155
156
157
158
159
                case '@':   /* EXEC Directory for DOS/OS2/Win32, blank for Unix */
#ifndef __unix__
                    strcat(cmd,scfg.exec_dir);
#endif
                    break;
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
				case '#':   /* Node number (same as SBBSNNUM environment var) */
					sprintf(str,"%d",scfg.node_num);
					strcat(cmd,str);
					break;
				case '%':   /* %% for percent sign */
					strcat(cmd,"%");
					break;
				default:    /* unknown specification */
					break; 
			}
			j=strlen(cmd); 
		}
		else
			cmd[j++]=instr[i]; 
	}
	cmd[j]=0;

	return(cmd);
}

/****************************************************************************/
/* Updates dstst.dab file													*/
/****************************************************************************/
void updatestats(ulong size)
{
    char	str[MAX_PATH+1];
    int		file;
deuce's avatar
64-bit    
deuce committed
187
	uint32_t	l;
188

189
	sprintf(str,"%sdsts.dab",scfg.ctrl_dir);
190
	if((file=nopen(str,O_RDWR|O_BINARY))==-1) {
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
		printf("ERR_OPEN %s\n",str);
		return; 
	}
	lseek(file,20L,SEEK_SET);	/* Skip timestamp, logons and logons today */
	read(file,&l,4);			/* Uploads today		 */
	l++;
	lseek(file,-4L,SEEK_CUR);
	write(file,&l,4);
	read(file,&l,4);			/* Upload bytes today	 */
	l+=size;
	lseek(file,-4L,SEEK_CUR);
	write(file,&l,4);
	close(file);
}

void addlist(char *inpath, file_t f, uint dskip, uint sskip)
{
208
209
210
211
212
213
	char str[MAX_PATH+1];
	char tmp[MAX_PATH+1];
	char fname[MAX_PATH+1];
	char listpath[MAX_PATH+1];
	char filepath[MAX_PATH+1];
	char curline[256],nextline[256];
214
	char *p;
215
	char ext[1024],tmpext[513];
216
217
	int i,file;
	long l;
218
	BOOL exist;
219
220
221
222
223
224
225
226
227
228
	FILE *stream;
	DIR*	dir;
	DIRENT*	dirent;

	if(mode&SEARCH_DIR) {
		strcpy(str,cur_altpath ? scfg.altpath[cur_altpath-1] : scfg.dir[f.dir]->path);
		printf("Searching %s\n\n",str);
		dir=opendir(str);

		while(dir!=NULL && (dirent=readdir(dir))!=NULL) {
rswindell's avatar
rswindell committed
229
230
231
			sprintf(tmp,"%s%s"
				,cur_altpath ? scfg.altpath[cur_altpath-1] : scfg.dir[f.dir]->path
				,dirent->d_name);
232
233
234
235
236
237
238
			if(isdir(tmp))
				continue;
#ifdef _WIN32
			GetShortPathName(tmp, filepath, sizeof(filepath));
#else
			strcpy(filepath,tmp);
#endif
239
240
241
			f.misc=0;
			f.desc[0]=0;
			f.cdt=flength(filepath);
242
			padfname(getfname(filepath),f.name);
deuce's avatar
deuce committed
243
			printf("%s  %10"PRIu32"  %s\n"
244
				,f.name,f.cdt,unixtodstr(&scfg,(time32_t)fdate(filepath),str));
245
246
247
248
249
250
			exist=findfile(&scfg,f.dir,f.name);
			if(exist) {
				if(mode&NO_UPDATE)
					continue;
				getfileixb(&scfg,&f);
				if(mode&ULDATE_ONLY) {
251
					f.dateuled=time32(NULL);
252
253
254
255
256
257
					update_uldate(&scfg, &f);
					continue; 
				} 
			}

			if(mode&FILE_DATE) {		/* get the file date and put into desc */
258
				unixtodstr(&scfg,(time32_t)fdate(filepath),f.desc);
259
260
261
262
				strcat(f.desc,"  "); 
			}

			if(mode&TODAYS_DATE) {		/* put today's date in desc */
263
				unixtodstr(&scfg,time32(NULL),f.desc);
264
265
266
267
268
				strcat(f.desc,"  "); 
			}

			if(mode&FILE_ID) {
				for(i=0;i<scfg.total_fextrs;i++)
rswindell's avatar
rswindell committed
269
					if(!stricmp(scfg.fextr[i]->ext,f.name+9) && chk_ar(&scfg,scfg.fextr[i]->ar,/* user: */NULL, /* client: */NULL))
270
271
272
						break;
				if(i<scfg.total_fextrs) {
					sprintf(tmp,"%sFILE_ID.DIZ",scfg.temp_dir);
273
					removecase(tmp);
274
					system(mycmdstr(scfg.fextr[i]->cmd,filepath,"FILE_ID.DIZ",NULL));
275
276
					if(!fexistcase(tmp)) {
						sprintf(tmp,"%sDESC.SDI",scfg.temp_dir);
277
						removecase(tmp);
278
						system(mycmdstr(scfg.fextr[i]->cmd,filepath,"DESC.SDI",NULL)); 
279
280
						fexistcase(tmp);
					}
281
					if((file=nopen(tmp,O_RDONLY|O_BINARY))!=-1) {
282
283
284
						memset(ext,0,513);
						read(file,ext,512);
						for(i=512;i;i--)
285
							if(ext[i-1]>' ' || ext[i-1]<0)
286
287
288
								break;
						ext[i]=0;
						if(mode&ASCII_ONLY)
289
							strip_exascii(ext, ext);
290
291
						if(!(mode&KEEP_DESC)) {
							sprintf(tmpext,"%.256s",ext);
292
							prep_desc(tmpext);
293
							for(i=0;tmpext[i];i++)
294
								if(isalpha((uchar)tmpext[i]))
295
296
									break;
							sprintf(f.desc,"%.*s",LEN_FDESC,tmpext+i);
297
							for(i=0;(f.desc[i]>=' ' || f.desc[i]<0) && i<LEN_FDESC;i++)
298
299
300
301
302
303
304
305
								;
							f.desc[i]=0; }
						close(file);
						f.misc|=FM_EXTDESC; 
					} 
				} 
			}

306
			f.dateuled=time32(NULL);
307
			f.altpath=cur_altpath;
308
			prep_desc(f.desc);
309
			if(mode&ASCII_ONLY)
310
				strip_exascii(f.desc, f.desc);
311
312
313
			if(exist) {
				putfiledat(&scfg,&f);
				if(!(mode&NO_NEWDATE))
314
315
					update_uldate(&scfg, &f); 
			}
316
317
318
319
			else
				addfiledat(&scfg,&f);
			if(f.misc&FM_EXTDESC) {
				truncsp(ext);
320
321
				putextdesc(&scfg,f.dir,f.datoffset,ext); 
			}
322
323
324
325
326
327
328
329
330
331
332
			if(mode&UL_STATS)
				updatestats(f.cdt);
			files++; 
		}
		if(dir!=NULL)
			closedir(dir);
		return; 
	}


	strcpy(listpath,inpath);
333
	fexistcase(listpath);
334
	if((stream=fopen(listpath,"r"))==NULL) {
335
336
		fprintf(stderr,"Error %d (%s) opening %s\n"
			,errno,strerror(errno),listpath);
337
338
		sprintf(listpath,"%s%s",cur_altpath ? scfg.altpath[cur_altpath-1]
				: scfg.dir[f.dir]->path,inpath);
339
		fexistcase(listpath);
340
		if((stream=fopen(listpath,"r"))==NULL) {
341
342
			printf("Can't open: %s\n"
				   "        or: %s\n",inpath,listpath);
343
344
345
			return; 
		} 
	}
346
347
348
349
350
351
352
353
354
355
356
357

	printf("Adding %s to %s %s\n\n"
		,listpath,scfg.lib[scfg.dir[f.dir]->lib]->sname,scfg.dir[f.dir]->sname);

	fgets(nextline,255,stream);
	while(!feof(stream) && !ferror(stream)) {
		f.misc=0;
		f.desc[0]=0;
		strcpy(curline,nextline);
		nextline[0]=0;
		fgets(nextline,255,stream);
		truncsp(curline);
358
		if(curline[0]<=' ' || (mode&ASCII_ONLY && (uchar)curline[0]>=0x7e))
359
360
361
362
363
364
365
			continue;
		printf("%s\n",curline);
		strcpy(fname,curline);

		p=strchr(fname,'.');
		if(!p || p==fname || p>fname+8)    /* no dot or invalid dot location */
			continue;
366
		p=strchr(p,' ');
367
368
369
		if(p) *p=0;
		else				   /* no space after filename? */
			continue;
370
#if 0
371
		strupr(fname);
372
#endif
373
374
375
376
377
378
379
		strcpy(fname,unpadfname(fname,tmp));

		padfname(fname,f.name);
		if(strcspn(f.name,"\\/|<>+[]:=\";,")!=strlen(f.name))
			continue;

		for(i=0;i<12;i++)
380
			if(f.name[i]<' ' || (mode&ASCII_ONLY && (uchar)f.name[i]>0x7e))
381
382
383
384
385
386
387
388
389
390
				break;

		if(i<12)					/* Ctrl chars or EX-ASCII in filename? */
			continue;
		exist=findfile(&scfg,f.dir,f.name);
		if(exist) {
			if(mode&NO_UPDATE)
				continue;
			getfileixb(&scfg,&f);
			if(mode&ULDATE_ONLY) {
391
				f.dateuled=time32(NULL);
392
				update_uldate(&scfg, &f);
393
394
395
				continue; 
			} 
		}
396
397
398
399
400

		sprintf(filepath,"%s%s",cur_altpath ? scfg.altpath[cur_altpath-1]
			: scfg.dir[f.dir]->path,fname);

		if(mode&FILE_DATE) {		/* get the file date and put into desc */
401
			l=(time32_t)fdate(filepath);
402
			unixtodstr(&scfg,l,f.desc);
403
404
			strcat(f.desc,"  "); 
		}
405
406

		if(mode&TODAYS_DATE) {		/* put today's date in desc */
407
			l=time32(NULL);
408
			unixtodstr(&scfg,l,f.desc);
409
410
			strcat(f.desc,"  "); 
		}
411
412
413
414

		if(dskip && strlen(curline)>=dskip) p=curline+dskip;
		else {
			p++;
415
			while(*p==' ') p++; 
416
		}
417
418
419
		SAFECOPY(tmp,p);
		prep_desc(tmp);
		sprintf(f.desc+strlen(f.desc),"%.*s",(int)(LEN_FDESC-strlen(f.desc)),tmp);
420

421
		if(nextline[0]==' ' || strlen(p)>LEN_FDESC) {	/* ext desc */
422
423
424
			if(!(mode&NO_EXTEND)) {
				memset(ext,0,513);
				f.misc|=FM_EXTDESC;
425
426
				sprintf(ext,"%s\r\n",p); 
			}
427

428
			if(nextline[0]==' ') {
429
430
				strcpy(str,nextline);				   /* tack on to end of desc */
				p=str+dskip;
431
				while(*p>0 && *p<=' ') p++;
432
433
434
435
436
437
				i=LEN_FDESC-strlen(f.desc);
				if(i>1) {
					p[i-1]=0;
					truncsp(p);
					if(p[0]) {
						strcat(f.desc," ");
438
439
440
441
						strcat(f.desc,p); 
					} 
				} 
			}
442
443

			while(!feof(stream) && !ferror(stream) && strlen(ext)<512) {
444
				if(nextline[0]!=' ')
445
446
447
448
449
450
					break;
				truncsp(nextline);
				printf("%s\n",nextline);
				if(!(mode&NO_EXTEND)) {
					f.misc|=FM_EXTDESC;
					p=nextline+dskip;
451
					while(*p==' ') p++;
452
					strcat(ext,p);
453
454
					strcat(ext,"\r\n"); 
				}
455
				nextline[0]=0;
456
457
458
				fgets(nextline,255,stream); 
			} 
		}
459
460
461
462
463
464
465


		if(sskip) l=atol(fname+sskip);
		else {
			l=flength(filepath);
			if(l<1L) {
				printf("%s not found.\n",filepath);
466
467
468
				continue; 
			} 
		}
469
470
471

		if(mode&FILE_ID) {
			for(i=0;i<scfg.total_fextrs;i++)
rswindell's avatar
rswindell committed
472
				if(!stricmp(scfg.fextr[i]->ext,f.name+9) && chk_ar(&scfg,scfg.fextr[i]->ar,/* user: */NULL, /* client: */NULL))
473
474
475
					break;
			if(i<scfg.total_fextrs) {
				sprintf(tmp,"%sFILE_ID.DIZ",scfg.temp_dir);
476
				removecase(tmp);
477
				system(mycmdstr(scfg.fextr[i]->cmd,filepath,"FILE_ID.DIZ",NULL));
478
479
				if(!fexistcase(tmp)) {
					sprintf(tmp,"%sDESC.SDI",scfg.temp_dir);
480
					removecase(tmp);
481
					system(mycmdstr(scfg.fextr[i]->cmd,filepath,"DESC.SDI",NULL)); 
482
483
					fexistcase(tmp);
				}
484
				if((file=nopen(tmp,O_RDONLY|O_BINARY))!=-1) {
485
486
487
					memset(ext,0,513);
					read(file,ext,512);
					for(i=512;i;i--)
488
						if(ext[i-1]>' ' || ext[i-1]<0)
489
490
491
							break;
					ext[i]=0;
					if(mode&ASCII_ONLY)
492
						strip_exascii(ext, ext);
493
494
					if(!(mode&KEEP_DESC)) {
						sprintf(tmpext,"%.256s",ext);
495
						prep_desc(tmpext);
496
						for(i=0;tmpext[i];i++)
497
							if(isalpha((uchar)tmpext[i]))
498
499
								break;
						sprintf(f.desc,"%.*s",LEN_FDESC,tmpext+i);
500
						for(i=0;(f.desc[i]>=' ' || f.desc[i]<0) && i<LEN_FDESC;i++)
501
							;
502
503
						f.desc[i]=0; 
					}
504
505
506
507
508
509
510
					close(file);
					f.misc|=FM_EXTDESC; 
				} 
			} 
		}

		f.cdt=l;
511
		f.dateuled=time32(NULL);
512
		f.altpath=cur_altpath;
513
		prep_desc(f.desc);
514
		if(mode&ASCII_ONLY)
515
			strip_exascii(f.desc, f.desc);
516
517
518
		if(exist) {
			putfiledat(&scfg,&f);
			if(!(mode&NO_NEWDATE))
519
520
				update_uldate(&scfg, &f); 
		}
521
522
523
524
		else
			addfiledat(&scfg,&f);
		if(f.misc&FM_EXTDESC) {
			truncsp(ext);
525
526
			putextdesc(&scfg,f.dir,f.datoffset,ext); 
		}
527
528
529

		if(mode&UL_STATS)
			updatestats(l);
530
531
		files++; 
	}
532
533
534
	fclose(stream);
	if(mode&DEL_LIST && !(mode&SYNC_LIST)) {
		printf("\nDeleting %s\n",listpath);
535
536
		remove(listpath); 
	}
537
538
539
540
541

}

void synclist(char *inpath, int dirnum)
{
542
	char	str[1024];
543
544
545
	char	fname[MAX_PATH+1];
	char	listpath[MAX_PATH+1];
	uchar*	ixbbuf;
546
	char*	p;
547
548
549
550
	int		i,file,found;
	long	l,m,length;
	FILE*	stream;
	file_t	f;
551

552
	sprintf(str,"%s%s.ixb",scfg.dir[dirnum]->data_dir,scfg.dir[dirnum]->code);
553
	if((file=nopen(str,O_RDONLY|O_BINARY))==-1) {
554
		printf("ERR_OPEN %s\n",str);
555
556
		return; 
	}
557
558
559
	length=filelength(file);
	if(length%F_IXBSIZE) {
		close(file);
rswindell's avatar
rswindell committed
560
		printf("ERR_LEN (%ld) of %s\n",length,str);
561
562
		return; 
	}
deuce's avatar
deuce committed
563
	if((ixbbuf=(uchar *)malloc(length))==NULL) {
564
565
		close(file);
		printf("ERR_ALLOC %s\n",str);
566
567
		return; 
	}
568
569
	if(lread(file,ixbbuf,length)!=length) {
		close(file);
deuce's avatar
deuce committed
570
		free((char *)ixbbuf);
571
		printf("ERR_READ %s\n",str);
572
573
		return; 
	}
574
575
576
	close(file);

	strcpy(listpath,inpath);
577
	if((stream=fopen(listpath,"r"))==NULL) {
578
579
		sprintf(listpath,"%s%s",cur_altpath ? scfg.altpath[cur_altpath-1]
				: scfg.dir[dirnum]->path,inpath);
580
		if((stream=fopen(listpath,"r"))==NULL) {
581
582
			printf("Can't open: %s\n"
				   "        or: %s\n",inpath,listpath);
583
584
585
			return; 
		} 
	}
586
587
588
589
590
591
592
593

	printf("\nSynchronizing %s with %s %s\n\n"
		,listpath,scfg.lib[scfg.dir[dirnum]->lib]->sname,scfg.dir[dirnum]->sname);

	for(l=0;l<length;l+=F_IXBSIZE) {
		m=l;
		for(i=0;i<12 && l<length;i++)
			if(i==8)
594
				str[i]=ixbbuf[m]>' ' ? '.' : ' ';
595
596
597
598
599
600
601
602
603
604
			else
				str[i]=ixbbuf[m++]; 	/* Turns FILENAMEEXT into FILENAME.EXT */
		str[i]=0;
		unpadfname(str,fname);
		rewind(stream);
		found=0;
		while(!found) {
			if(!fgets(str,1000,stream))
				break;
			truncsp(str);
605
			p=strchr(str,' ');
606
607
			if(p) *p=0;
			if(!stricmp(str,fname))
608
609
				found=1; 
		}
610
611
612
613
614
615
616
617
618
		if(found)
			continue;
		padfname(fname,f.name);
		printf("%s not found in list - ",f.name);
		f.dir=dirnum;
		f.datoffset=ixbbuf[m]|((long)ixbbuf[m+1]<<8)|((long)ixbbuf[m+2]<<16);
		getfiledat(&scfg,&f);
		if(f.opencount) {
			printf("currently OPEN by %u users\n",f.opencount);
619
620
			continue; 
		}
621
		removefiledat(&scfg,&f);
622
		if(remove(getfilepath(&scfg,&f,str)))
623
624
			printf("Error removing %s\n",str);
		removed++;
625
626
		printf("Removed from database\n"); 
	}
627
628
629

	if(mode&DEL_LIST) {
		printf("\nDeleting %s\n",listpath);
630
631
		remove(listpath); 
	}
632
633
}

634
char *usage="\nusage: addfiles code [.alt_path] [-opts] +list "
635
			 "[desc_off] [size_off]"
636
	"\n   or: addfiles code [.alt_path] [-opts] file "
637
638
		"\"description\"\n"
	"\navailable opts:"
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
	"\n      -a         import ASCII only (no extended ASCII)"
	"\n      -b         synchronize database with file list (use with caution)"
	"\n      -c         do not remove extra spaces from file description"
	"\n      -d         delete list after import"
	"\n      -e         do not import extended descriptions"
	"\n      -f         include file date in descriptions"
	"\n      -t         include today's date in descriptions"
	"\n      -i         include added files in upload statistics"
	"\n      -n         do not update information for existing files"
	"\n      -o         update upload date only for existing files"
	"\n      -u         do not update upload date for existing files"
	"\n      -z         check for and import FILE_ID.DIZ and DESC.SDI"
	"\n      -k         keep original short description (not DIZ)"
	"\n      -s         search directory for files (no file list)"
	"\n      -l <lib>   specify library (short name) to Auto-ADD"
	"\n      -x <name>  specify uploader's user name (may require quotes)"
655
	"\n"
656
657
	"\nAuto-ADD:   use - in place of code for Auto-ADD of FILES.BBS"
	"\n            use -filename to Auto-ADD a different filename"
658
	"\n            use -l \"libname\" to only Auto-ADD files to a specific library"
659
660
661
662
663
664
665
666
667
	"\n"
	;

/*********************/
/* Entry point (duh) */
/*********************/
int main(int argc, char **argv)
{
	char error[512];
668
	char revision[16];
669
670
	char str[MAX_PATH+1];
	char tmp[MAX_PATH+1];
671
672
	char *p;
	char exist,listgiven=0,namegiven=0,ext[513]
673
		,auto_name[MAX_PATH+1]="FILES.BBS";
674
	int i,j,file;
675
	uint desc_offset=0, size_offset=0;
676
677
678
	long l;
	file_t	f;

679
	sscanf("$Revision$", "%*s %s", revision);
680

681
	fprintf(stderr,"\nADDFILES v%s-%s (rev %s) - Adds Files to Synchronet "
682
683
684
		"Filebase\n"
		,ADDFILES_VER
		,PLATFORM_DESC
685
		,revision
686
687
688
689
		);

	if(argc<2) {
		printf(usage);
690
691
		return(1); 
	}
692
693
694
695
696
697
698
699

	p=getenv("SBBSCTRL");
	if(p==NULL) {
		printf("\nSBBSCTRL environment variable not set.\n");
		printf("\nExample: SET SBBSCTRL=/sbbs/ctrl\n");
		exit(1); 
	}

700
701
	memset(&scfg,0,sizeof(scfg));
	scfg.size=sizeof(scfg);
702
	SAFECOPY(scfg.ctrl_dir,p);
703
704
705
706
707

	if(chdir(scfg.ctrl_dir)!=0)
		fprintf(stderr,"!ERROR changing directory to: %s", scfg.ctrl_dir);

	printf("\nLoading configuration files from %s\n",scfg.ctrl_dir);
708
709
710
711
	if(!load_cfg(&scfg,NULL,TRUE,error)) {
		fprintf(stderr,"!ERROR loading configuration files: %s\n",error);
		exit(1);
	}
rswindell's avatar
rswindell committed
712
713
	SAFECOPY(scfg.temp_dir,"../temp");
	prep_dir(scfg.ctrl_dir, scfg.temp_dir, sizeof(scfg.temp_dir));
714

715
716
717
718
719
	if(argv[1][0]=='*' || argv[1][0]=='-') {
		if(argv[1][1]=='?') {
			printf(usage);
			exit(0);
		}
720
		if(argv[1][1])
721
			SAFECOPY(auto_name,argv[1]+1);
722
		mode|=AUTO_ADD;
723
724
		i=0; 
	} else {
725
		if(!isalnum((uchar)argv[1][0]) && argc==2) {
726
727
728
729
			printf(usage);
			return(1); 
		}

730
		for(i=0;i<scfg.total_dirs;i++)
731
			if(!stricmp(scfg.dir[i]->code,argv[1])) /* use matchmatchi() instead? */
732
733
734
735
				break;

		if(i>=scfg.total_dirs) {
			printf("Directory code '%s' not found.\n",argv[1]);
736
737
738
			exit(1); 
		} 
	}
739
740
741
742
743
744

	memset(&f,0,sizeof(file_t));
	f.dir=i;
	strcpy(f.uler,"-> ADDFILES <-");

	for(j=2;j<argc;j++) {
745
746
		if(argv[j][0]=='*')     /* set the uploader name (legacy) */
			SAFECOPY(f.uler,argv[j]+1);
747
748
749
750
		else 
			if(argv[j][0]=='-'
			|| argv[j][0]=='/'
			) {      /* options */
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
			for(i=1;argv[j][i];i++)
				switch(toupper(argv[j][i])) {
					case 'A':
						mode|=ASCII_ONLY;
						break;
					case 'B':
						mode|=(SYNC_LIST|NO_UPDATE);
						break;
					case 'C':
						mode|=KEEP_SPACE;
						break;
					case 'D':
						mode|=DEL_LIST;
						break;
					case 'E':
						mode|=NO_EXTEND;
						break;
					case 'I':
						mode|=UL_STATS;
						break;
771
772
773
774
775
776
777
778
779
					case 'L':
						j++;
						if(argv[j]==NULL) {
							printf(usage);
							return(-1);
						}
						SAFECOPY(lib,argv[j]);
						i=strlen(argv[j])-1;
						break;
780
781
782
783
784
785
786
787
788
					case 'X':
						j++;
						if(argv[j]==NULL) {
							printf(usage);
							return(-1);
						}
						SAFECOPY(f.uler,argv[j]);
						i=strlen(argv[j])-1;
						break;
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
					case 'Z':
						mode|=FILE_ID;
						break;
					case 'K':
						mode|=KEEP_DESC;	 /* Don't use DIZ for short desc */
						break;
					case 'N':
						mode|=NO_UPDATE;
						break;
					case 'O':
						mode|=ULDATE_ONLY;
						break;
					case 'U':
						mode|=NO_NEWDATE;
						break;
					case 'F':
						mode|=FILE_DATE;
						break;
					case 'T':
						mode|=TODAYS_DATE;
						break;
					case 'S':
						mode|=SEARCH_DIR;
						break;
					default:
						printf(usage);
815
816
817
						return(1); 
			} 
		}
818
		else if(isdigit((uchar)argv[j][0])) {
819
820
821
822
823
824
			if(desc_offset==0)
				desc_offset=atoi(argv[j]);
			else
				size_offset=atoi(argv[j]);
			continue;
		}
825
826
		else if(argv[j][0]=='+') {      /* filelist - FILES.BBS */
			listgiven=1;
rswindell's avatar
rswindell committed
827
			if(argc > j+1
828
				&& isdigit((uchar)argv[j+1][0])) { /* skip x characters before description */
rswindell's avatar
rswindell committed
829
				if(argc > j+2
830
					&& isdigit((uchar)argv[j+2][0])) { /* skip x characters before size */
831
					addlist(argv[j]+1,f,atoi(argv[j+1]),atoi(argv[j+2]));
832
833
					j+=2; 
				}
834
835
				else {
					addlist(argv[j]+1,f,atoi(argv[j+1]),0);
836
837
838
					j++; 
				} 
			}
839
840
841
			else
				addlist(argv[j]+1,f,0,0);
			if(mode&SYNC_LIST)
842
843
				synclist(argv[j]+1,f.dir); 
		}
844
845
846
847
		else if(argv[j][0]=='.') {      /* alternate file path */
			cur_altpath=atoi(argv[j]+1);
			if(cur_altpath>scfg.altpaths) {
				printf("Invalid alternate path.\n");
848
849
850
				exit(1); 
			} 
		}
851
852
853
854
		else {
			namegiven=1;
			padfname(argv[j],f.name);
			f.desc[0]=0;
855
#if 0
856
			strupr(f.name);
857
#endif
858
859
			if(j+1==argc) {
				printf("%s no description given.\n",f.name);
860
861
				continue; 
			}
862
863
864
			sprintf(str,"%s%s",cur_altpath ? scfg.altpath[cur_altpath-1]
				: scfg.dir[f.dir]->path,argv[j]);
			if(mode&FILE_DATE)
865
				sprintf(f.desc,"%s  ",unixtodstr(&scfg,(time32_t)fdate(str),tmp));
866
			if(mode&TODAYS_DATE)
867
				sprintf(f.desc,"%s  ",unixtodstr(&scfg,time32(NULL),tmp));
rswindell's avatar
rswindell committed
868
			sprintf(tmp,"%.*s",(int)(LEN_FDESC-strlen(f.desc)),argv[++j]);
869
870
871
872
			strcpy(f.desc,tmp);
			l=flength(str);
			if(l==-1) {
				printf("%s not found.\n",str);
873
874
				continue; 
			}
875
876
877
878
879
880
			exist=findfile(&scfg,f.dir,f.name);
			if(exist) {
				if(mode&NO_UPDATE)
					continue;
				getfileixb(&scfg,&f);
				if(mode&ULDATE_ONLY) {
881
					f.dateuled=time32(NULL);
882
					update_uldate(&scfg, &f);
883
884
885
					continue; 
				} 
			}
886
			f.cdt=l;
887
			f.dateuled=time32(NULL);
888
			f.altpath=cur_altpath;
889
			prep_desc(f.desc);
890
			if(mode&ASCII_ONLY)
891
				strip_exascii(f.desc, f.desc);
deuce's avatar
deuce committed
892
			printf("%s %7"PRIu32" %s\n",f.name,f.cdt,f.desc);
893
894
			if(mode&FILE_ID) {
				for(i=0;i<scfg.total_fextrs;i++)
rswindell's avatar
rswindell committed
895
					if(!stricmp(scfg.fextr[i]->ext,f.name+9) && chk_ar(&scfg,scfg.fextr[i]->ar,/* user: */NULL, /* client: */NULL))
896
897
898
						break;
				if(i<scfg.total_fextrs) {
					sprintf(tmp,"%sFILE_ID.DIZ",scfg.temp_dir);
899
					removecase(tmp);
900
					system(mycmdstr(scfg.fextr[i]->cmd,str,"FILE_ID.DIZ",NULL));
901
					if(!fexistcase(tmp)) {
902
						sprintf(tmp,"%sDESC.SDI",scfg.temp_dir);
903
						removecase(tmp);
904
						system(mycmdstr(scfg.fextr[i]->cmd,str,"DESC.SDI",NULL)); 
905
906
						fexistcase(tmp);
					}
907
					if((file=nopen(tmp,O_RDONLY|O_BINARY))!=-1) {
908
909
910
911
						memset(ext,0,513);
						read(file,ext,512);
						if(!(mode&KEEP_DESC)) {
							sprintf(f.desc,"%.*s",LEN_FDESC,ext);
912
							for(i=0;(f.desc[i]>=' ' || f.desc[i]<0) && i<LEN_FDESC;i++)
913
								;
914
915
							f.desc[i]=0; 
						}
916
						close(file);
917
918
919
920
						f.misc|=FM_EXTDESC; 
					} 
				} 
			}
921
922
923
			if(exist) {
				putfiledat(&scfg,&f);
				if(!(mode&NO_NEWDATE))
924
925
					update_uldate(&scfg, &f); 
			}
926
927
928
929
930
931
932
933
			else
				addfiledat(&scfg,&f);

			if(f.misc&FM_EXTDESC)
				putextdesc(&scfg,f.dir,f.datoffset,ext);

			if(mode&UL_STATS)
				updatestats(l);
934
935
936
			files++; 
		} 
	}
937
938
939

	if(mode&AUTO_ADD) {
		for(i=0;i<scfg.total_dirs;i++) {
940
941
			if(lib[0] && stricmp(scfg.lib[scfg.dir[i]->lib]->sname,lib))
				continue;
942
943
944
945
			if(scfg.dir[i]->misc&DIR_NOAUTO)
				continue;
			f.dir=i;
			if(mode&SEARCH_DIR) {
946
				addlist("",f,desc_offset,size_offset);
947
948
				continue; 
			}
949
			sprintf(str,"%s%s.lst",scfg.dir[f.dir]->path,scfg.dir[f.dir]->code);
950
			if(fexistcase(str) && flength(str)>0L) {
951
				printf("Auto-adding %s\n",str);
952
				addlist(str,f,desc_offset,size_offset);
953
954
				if(mode&SYNC_LIST)
					synclist(str,i);
955
956
				continue; 
			}
957
			sprintf(str,"%s%s",scfg.dir[f.dir]->path,auto_name);
958
			if(fexistcase(str) && flength(str)>0L) {
959
				printf("Auto-adding %s\n",str);
960
				addlist(str,f,desc_offset,size_offset);
961
962
				if(mode&SYNC_LIST)
					synclist(str,i);
963
964
965
966
				continue; 
			} 
		} 
	}
967
968
969

	else {
		if(!listgiven && !namegiven) {
970
			sprintf(str,"%s%s.lst",scfg.dir[f.dir]->path, scfg.dir[f.dir]->code);
971
			if(!fexistcase(str) || flength(str)<=0L)
972
				sprintf(str,"%s%s",scfg.dir[f.dir]->path, auto_name);
973
			addlist(str,f,desc_offset,size_offset);
974
			if(mode&SYNC_LIST)
975
976
977
				synclist(str,f.dir); 
		} 
	}
978
979
980
981
982
983
984
985

	printf("\n%lu file(s) added.",files);
	if(removed)
		printf("\n%lu files(s) removed.",removed);
	printf("\n");
	return(0);
}