- Notifications
You must be signed in to change notification settings - Fork46
/
Copy pathftpserver_ini.cpp
517 lines (480 loc) · 20.5 KB
/
ftpserver_ini.cpp
1
2
3
4
5
6
7
8
9
10
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
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
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
155
156
157
158
159
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
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
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
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
/*******************************************************************
*ftpserver.h
* DESCRIPTION: FTP服务配置的保存和读取
*
* AUTHOR:yyc
*
* http://hi.baidu.com/yycblog/home
*
*******************************************************************/
#include"rmtsvc.h"
bool ftpsvrEx :: readIni()
{
HKEY hKEY;
LPCTSTR lpRegPath ="Software\\Microsoft\\Windows\\CurrentVersion";
if(::RegOpenKeyEx(HKEY_LOCAL_MACHINE, lpRegPath,0, KEY_WRITE|KEY_READ, &hKEY)!=ERROR_SUCCESS)
returnfalse;
DWORD dwType=REG_SZ,dwSize=0;
char *pbuffer=NULL;
if(::RegQueryValueEx(hKEY,"ftpsets",NULL,&dwType,NULL,&dwSize)==ERROR_SUCCESS)
{
if( (pbuffer=newchar[dwSize]) )
::RegQueryValueEx(hKEY,"ftpsets",NULL,&dwType,(LPBYTE)pbuffer,&dwSize);
}
::RegCloseKey(hKEY);
if(pbuffer==NULL)returnfalse;
dwSize=cCoder::base64_decode(pbuffer,dwSize-1,pbuffer);
pbuffer[dwSize]=0;
parseIni(pbuffer,dwSize);
delete[] pbuffer;returntrue;
}
bool ftpsvrEx :: parseIni(char *pbuffer,long lsize)
{
if(pbuffer==NULL)returnfalse;
if(lsize<=0) lsize=strlen(pbuffer);
if(strncasecmp(pbuffer,"base64",6)==0)
lsize=cCoder::base64_decode(pbuffer+6,lsize-6,pbuffer);
//RW_LOG_DEBUG("read FTP param.........\r\n%s\r\n",pbuffer);
//开始解析Ini文件
constchar *pstart=pbuffer;
constchar *ptr=strchr(pstart,'\r');
while(true){
if(ptr) *(char *)ptr=0;
if(pstart[0]!='!')//不解释注释行
{
if(strncasecmp(pstart,"sets",5)==0)//设置本服务的信息
docmd_sets(pstart+5);
elseif(strncasecmp(pstart,"ssls",5)==0)//设置本服务ssl信息
docmd_ssls(pstart+5);
elseif(strncasecmp(pstart,"ftps",5)==0)//设置ftp服务的欢迎信息
docmd_ftps(pstart+5);
elseif(strncasecmp(pstart,"iprules",8)==0)//设置访问此服务IP过滤规则
docmd_iprules(pstart+8);
elseif(strncasecmp(pstart,"user",5)==0)//添加帐号
docmd_user(pstart+5);
elseif(strncasecmp(pstart,"vpath",6)==0)//添加帐号的虚目录
docmd_vpath(pstart+6);
}
if(ptr==NULL)break;
*(char *)ptr='\r'; pstart=ptr+1;
if(*pstart=='\n') pstart++;
ptr=strchr(pstart,'\r');
}//?while
returntrue;
}
bool ftpsvrEx :: saveAsstring(std::string &strini)
{
char buf[256];
int len=sprintf(buf,"sets autorun=%d port=%d bindip=%s maxuser=%d\r\n",
(m_settings.autoStart)?1:0,m_settings.svrport,m_settings.bindip.c_str(),m_settings.maxUsers);
strini.append(buf,len);
len=sprintf(buf,"ssls enable=%s\r\n",(m_settings.ifSSLsvr)?"true":"false");
strini.append(buf,len);
len=sprintf(buf,"ftps dataport=%d:%d logevent=%d tips=\"",m_settings.dataportB,m_settings.dataportE,
m_settings.logEvent);
strini.append(buf,len);
constchar *ptrBegin=m_settings.tips.c_str();
constchar *ptr=strstr(ptrBegin,"\r\n");
while(true)
{
if(ptr)
strini.append(ptrBegin,ptr-ptrBegin);
else{ strini.append(ptrBegin);break; }
strini.append("\\n");
ptrBegin=ptr+2;
ptr=strstr(ptrBegin,"\r\n");
}
strini.append("\"\r\n");
//-------------------------保存帐号信息------------------------
std::map<std::string,TFTPUser>::iterator it=m_userlist.begin();
for(;it!=m_userlist.end();it++)
{
TFTPUser &ftpuser=(*it).second;
len=sprintf(buf,"user account=%s pswd=%s pswdmode=%d hiddenfile=%d maxlogin=%d forbid=%d",
ftpuser.username.c_str(),ftpuser.userpwd.c_str(),ftpuser.pswdmode,ftpuser.disphidden,
ftpuser.maxLoginusers,ftpuser.forbid);
strini.append(buf,len);
len=sprintf(buf," maxup=%d maxdw=%d maxupfile=%d maxdisksize=%d:%d",
ftpuser.maxupratio,ftpuser.maxdwratio,ftpuser.maxupfilesize,ftpuser.maxdisksize,ftpuser.curdisksize);
strini.append(buf,len);
if(ftpuser.limitedTime!=0)
{
structtm * ltime=localtime(&ftpuser.limitedTime);
len=sprintf(buf," expired=%04d-%02d-%02d",(1900+ltime->tm_year), ltime->tm_mon+1, ltime->tm_mday);
strini.append(buf,len);
}
std::pair<std::string,long> &p=ftpuser.dirAccess["/"];
len=sprintf(buf," access=%d root=\"%s\" desc=\"",p.second,p.first.c_str());
strini.append(buf,len);
strini+=ftpuser.userdesc; strini.append("\"\r\n");
len=sprintf(buf,"iprules account=%s access=%d",ftpuser.username.c_str(),ftpuser.ipaccess);
strini.append(buf,len);
len=sprintf(buf," ipaddr=%s\r\n",ftpuser.ipRules.c_str());
strini.append(buf,len);
std::map<std::string,std::pair<std::string,long> >::iterator it1=ftpuser.dirAccess.begin();
it1++;
for(;it1!=ftpuser.dirAccess.end();it1++)
{
std::pair<std::string,long> &p1=(*it1).second;
len=sprintf(buf,"vpath account=%s vdir=%s access=%d",ftpuser.username.c_str(),(*it1).first.c_str(),p1.second);
strini.append(buf,len);
len=sprintf(buf," rdir=\"%s\"\r\n",p1.first.c_str());
strini.append(buf,len);
}
}//?for(
//RW_LOG_DEBUG("save FTP param.........\r\n%s\r\n",strini.c_str());
returntrue;
}
bool ftpsvrEx :: saveIni()
{
std::string strini;
saveAsstring(strini);
//写注册表--------------------------------------
HKEY hKEY;
LPCTSTR lpRegPath ="Software\\Microsoft\\Windows\\CurrentVersion";
if(::RegOpenKeyEx(HKEY_LOCAL_MACHINE, lpRegPath,0, KEY_WRITE|KEY_READ, &hKEY)==ERROR_SUCCESS)
{
long l=cCoder::Base64EncodeSize(strini.length());
char *pbuf=newchar[l];
if(pbuf){
l=cCoder::base64_encode ((char *)strini.c_str(),strini.length(),pbuf);
::RegSetValueEx(hKEY,"ftpsets",NULL,REG_SZ, (LPBYTE)pbuf,l+1);
delete[] pbuf;
}
::RegCloseKey(hKEY);
}
returntrue;
}
//设置本服务的信息
//命令格式:
//sets [autorun=0|1] [port=<服务端口>] [bindip=<本服务绑定的本机IP>] [maxuser=<最大并发连接数>]
//port=<服务端口> : 设置服务端口,如果不设置则默认为21
//bindip=<本服务绑定的本机IP> : 设置本服务绑定的本机IP,如果不设置则默认绑定本机所有IP
//maxuser=<最大并发连接数> : 设置本服务的最大并发用户连接数,如果不设置此项则为不限制
void ftpsvrEx :: docmd_sets(constchar *strParam)
{
std::map<std::string,std::string> maps;
if(splitString(strParam,'',maps)<=0)return;
std::map<std::string,std::string>::iterator it;
if( (it=maps.find("autorun"))!=maps.end())
{//设置服务的端口
if(atoi((*it).second.c_str())==1)
m_settings.autoStart=true;
else m_settings.autoStart=false;
}
if( (it=maps.find("port"))!=maps.end())
{//设置服务的端口
m_settings.svrport=atoi((*it).second.c_str());
if(m_settings.svrport<0) m_settings.svrport=0;
}
if( (it=maps.find("bindip"))!=maps.end())
{//设置服务绑定IP
m_settings.bindip=(*it).second;
}
if( (it=maps.find("maxuser"))!=maps.end())
{//设置服务的最大并发数
m_settings.maxUsers=atoi((*it).second.c_str());
if(m_settings.maxUsers<0) m_settings.maxUsers=0;
}
return;
}
//设置本服务ssl信息
//命令格式:
//ssls [enable=<true|false>] [cacert=<SSL证书文件>] [cakey=<SSL私钥文件>] [capwd=<私钥密码>]
//enable=<true|false> : 指示服务是否为SSL加密服务,如果设置为true则为SSL加密服务,默认为非SSL加密服务
//下面三项指示SSL加密协商的证书、私钥和私钥密码,如果证书或私钥任意一个不写则用程序内置的证书和私钥进行SSL加密协商
//cacert=<SSL证书文件>: 指定SSL加密协商的证书文件名,证书为.pem格式的证书。可指定相对或绝对路径
//cakey=<SSL私钥文件> : 指定SSL加密协商的私钥文件名,私钥为.pem格式的私钥。可指定相对或绝对路径
//capwd=<私钥密码> : 指定的私钥的密码
void ftpsvrEx :: docmd_ssls(constchar *strParam)
{
#ifdef _SURPPORT_OPENSSL_
std::map<std::string,std::string> maps;
if(splitString(strParam,'',maps)<=0)return;
std::string strCert,strKey,strPwd;
std::map<std::string,std::string>::iterator it;
if( (it=maps.find("cacert"))!=maps.end())//设置SSL证书
strCert=(*it).second;
if( (it=maps.find("cakey"))!=maps.end())//设置SSL证书私钥
strKey=(*it).second;
if( (it=maps.find("capwd"))!=maps.end())//设置SSL证书私钥密码
strPwd=(*it).second;
if(strCert!="" && strKey!="")
{
getAbsolutfilepath(strCert);
getAbsolutfilepath(strKey);
setCacert(strCert.c_str(),strKey.c_str(),strPwd.c_str(),false);//使用用户指定的证书
}
elsesetCacert(NULL,NULL,NULL,true);//使用内置的证书
if( (it=maps.find("enable"))!=maps.end())//设置SSL证书
if(strcasecmp((*it).second.c_str(),"true")==0)
m_settings.ifSSLsvr=true;
else m_settings.ifSSLsvr=false;
#endif
return;
}
//设置FTP服务的欢迎信息和绑定数据传输端口
//命令格式:
//ftps [dataport=<端口起始:端口结束>] [tips="<欢迎词信息>"] [logevent=LOGIN|LOGOUT|DELETE|RMD|UPLOAD|DWLOAD|SITE]
//dataport=<端口起始:端口结束>: 设置本FTP服务的PASV数据传输端口,如果不指定则由系统随机分配,否则在你指定的区间随机分配
//[端口起始:端口结束]范围为闭区间,端口为0则代表不限,例如[500,0]代表分配的端口要>=500,[0,600]代表分配的端口要<=600
//tips="<欢迎词信息>" : 指定ftp客户端连接到服务后显示的欢迎信息,因为欢迎信息可能包含空格,因此最好用""括起
//可以写多行欢迎信息,每行的结尾必须以"\n"字符串结束。每行的起始用"220-"开头
//logevent=LOGIN|LOGOUT|DELETE|RMD|UPLOAD|DWLOAD|SITE : 指定记录哪些INFO事件
// 当loglevel记录日志级别等于DEBUG或INFO时指定要记录哪些事件,默认记录LOGIN,LOGOUT,SITE扩展命令事件
// 如要指定记录多个事件可用|连接。例如logevent=LOGIN|LOGOUT|DELETE
// LOGIN事件 -- 记录某个用户登录时间和IP
// LOGOUT事件 -- 记录某个用户离开时间和IP
// DELETE事件 -- 记录某个用户删除文件
// RMD事件 -- 记录某个用户删除目录
// UPLOAD事件 -- 记录某个用户上载文件
// DWLOAD事件 -- 记录某个用户下载文件
// SITE事件 -- 记录某个用户执行FTP扩展SITE命令(不包含SITE INFO)以及命令执行结果
//范例: ftps tips="220-欢迎访问XXXftp服务\n220-test line.\n"
void ftpsvrEx :: docmd_ftps(constchar *strParam)
{
if(strParam==NULL)return;
std::map<std::string,std::string> maps;
if(splitString(strParam,'',maps)<=0)return;
std::map<std::string,std::string>::iterator it;
if( (it=maps.find("dataport"))!=maps.end())
{//设置服务的数据传输端口范围
m_settings.dataportB=0;
m_settings.dataportE=0;
constchar *ptr=strchr((*it).second.c_str(),':');
if(ptr){
*(char *)ptr=0;
m_settings.dataportB=atoi((*it).second.c_str());
*(char *)ptr=':';
m_settings.dataportE=atoi(ptr+1);
}
else m_settings.dataportB=m_settings.dataportE=atoi((*it).second.c_str());
}
if( (it=maps.find("logevent"))!=maps.end())
{//设置要输出日志的事件
constchar *ptr=(*it).second.c_str();
m_settings.logEvent=atoi(ptr);
if(strstr(ptr,"LOGIN")) m_settings.logEvent|=FTP_LOGEVENT_LOGIN;
if(strstr(ptr,"LOGOUT")) m_settings.logEvent|=FTP_LOGEVENT_LOGOUT;
if(strstr(ptr,"DELETE")) m_settings.logEvent|=FTP_LOGEVENT_DELETE;
if(strstr(ptr,"RMD")) m_settings.logEvent|=FTP_LOGEVENT_RMD;
if(strstr(ptr,"UPLOAD")) m_settings.logEvent|=FTP_LOGEVENT_UPLOAD;
if(strstr(ptr,"DWLOAD")) m_settings.logEvent|=FTP_LOGEVENT_DWLOAD;
if(strstr(ptr,"SITE")) m_settings.logEvent|=FTP_LOGEVENT_SITE;
}
if( (it=maps.find("tips"))!=maps.end())
{//将"\n"替换为\r\n
constchar *ptr,*ptrStart=(*it).second.c_str();
while( (ptr=strstr(ptrStart,"\\n")) )
{
*(char *)ptr='\r';
*(char *)(ptr+1)='\n';
ptrStart=ptr+2;
}
m_settings.tips=(*it).second;
}
return;
}
//添加新的ftp帐号信息,如果帐号已存在则删除旧的,重新添加
//命令格式:
//user account=<ftp帐号> pswd=<帐号密码> root=<主目录> [pswdmode=<OTP_NONE|OTP_MD4|OTP_MD5>] [access=<对主目录的访问权限>] [hiddenfile=HIDE] [maxlogin=<限制同时最多登录人数>] [expired=<帐号有效期限>] [maxup=<最大上载流量>] [maxdw=<最大下载流量>] [maxupfile=<最大上载文件大小>] [maxdisksize=<最大可使用磁盘空间>[:<当前已使用磁盘空间>]]
//account=<ftp帐号> : 必须项. 要添加的ftp帐号。
//pswd=<帐号密码> : 必须项. 指定ftp帐号的密码,
//如果帐号密码等于""或空则意味着无需密码访问,只要帐号名对就可访问
//root=<主目录> : 必须项. 指定此ftp帐号的根/对应的主目录。主目录是运行ftp服务本地机的实际绝对路径
//如果主目录中包含空格则要用""将主目录括起
//如果主目录等于""或空则意味着此帐号可访问计算机所有磁盘目录
//pswdmode=<OTP_NONE|OTP_MD4|OTP_MD5> : 密码保护模式。OTP_NONE:明文传输密码
//OTP_MD4: otp S/Key md4加密密码传输. ftp客户端的密码保护应设置为md4或otp自动检测模式
//OTP_MD4: otp S/Key md5加密密码传输. ftp客户端的密码保护应设置为md5或otp自动检测模式
//如果不设置则默认是OTP_NONE,即密码明文传输。
//access=<对主目录的访问权限> : 指定此ftp帐号的主目录的访问权限。
//如果不设置则默认具有ACCESS_ALL访问权限。设置格式和含义如下
//<对主目录的访问权限> : <FILE_READ|FILE_WRITE|FILE_DEL|FILE_EXEC|DIR_LIST|DIR_MAKE|DIR_DEL|DIR_NOINHERIT>
//ACCESS_ALL=FILE_READ|FILE_WRITE|FILE_DEL|FILE_EXEC|DIR_LIST|DIR_MAKE|DIR_DEL
//FILE_READ : 允许读文件 FILE_WRITE : 允许写文件 FILE_DEL : 允许删除文件 FILE_EXEC : 允许执行文件
//DIR_LIST : 允许目录文件list DIR_MAKE : 允许创建目录 DIR_DEL : 允许删除目录
//DIR_NOINHERIT : 是否允许此虚目录对应的真实路径下的子目录继承用户指定的目录访问权限。如果用户指定了此项则禁止继承,
// 那么默认ftp用户是无法对其子目录进行任何的读写操作,要想对其下的子目录具有操作,必须将其下的子目录映射为虚目录单独指定访问权限
//注意:如果设定了FILE_EXEC ,则用户可通过扩展FTP命令EXEC,远程执行此虚目录下的文件。
//hiddenfile=HIDE : 不显示隐藏文件。如果不设置此项默认为显示隐藏文件/目录
//maxlogin=<限制同时最多登录人数> : 限制此ftp帐号的同时登录访问人数。
//如果不设置则默认为0,即不限制此帐号的同时登录人数
//expired=<帐号有效期限> : 限制此帐号的使用期限,格式YYYY-MM-DD
//如果不设置则帐号永不过期
//maxup=<最大上载流量> : 限制此ftp帐号的最大上载流量Kb/秒
//如果不设置则默认为0,即不限制此帐号的最大上载流量
//maxdw=<最大下载流量> : 限制此ftp帐号的最大下载流量Kb/秒
//如果不设置则默认为0,即不限制此帐号的最大下载流量
//maxupfile=<最大上载文件大小> : 限制此ftp帐号的最大上载文件的大小 KBytes
//如果不设置则默认为0,即不限制此帐号的最大上载文件大小
//maxdisksize=<最大可使用磁盘空间>[:<当前已使用磁盘空间>] : 限制此帐号最大可使用的磁盘空间KBytes
//<最大可使用磁盘空间>: KBytes,设置此帐号最大可使用的磁盘空间。如果不设置则 默认为0即不限制
//[:<当前已使用磁盘空间>] :KBytes,设置当前已使用的磁盘空间。如果不设置则默认为0,即没有使用
void ftpsvrEx :: docmd_user(constchar *strParam)
{
std::map<std::string,std::string> maps;
if(splitString(strParam,'',maps)<=0)return;
TFTPUser ftpuser;
std::map<std::string,std::string>::iterator it;
if( (it=maps.find("account"))!=maps.end() )
ftpuser.username=(*it).second;
if(ftpuser.username=="")return;
if( (it=maps.find("pswd"))!=maps.end())
ftpuser.userpwd=(*it).second;
if( (it=maps.find("desc"))!=maps.end())
ftpuser.userdesc=(*it).second;
std::pair<std::string,long>p("",0);
if( (it=maps.find("root"))!=maps.end())
p.first=(*it).second;
if( (it=maps.find("access"))!=maps.end() && (*it).second!="")
{
constchar *ptr=(*it).second.c_str();
long lAccess=atoi(ptr);
if(strstr(ptr,"FILE_READ")) lAccess|=FTP_ACCESS_FILE_READ;
if(strstr(ptr,"FILE_WRITE")) lAccess|=FTP_ACCESS_FILE_WRITE;
if(strstr(ptr,"FILE_DEL")) lAccess|=FTP_ACCESS_FILE_DELETE;
if(strstr(ptr,"FILE_EXEC")) lAccess|=FTP_ACCESS_FILE_EXEC;
if(strstr(ptr,"DIR_LIST")) lAccess|=FTP_ACCESS_DIR_LIST;
if(strstr(ptr,"DIR_MAKE")) lAccess|=FTP_ACCESS_DIR_CREATE;
if(strstr(ptr,"DIR_DEL")) lAccess|=FTP_ACCESS_DIR_DELETE;
if(strstr(ptr,"DIR_NOINHERIT")) lAccess|=FTP_ACCESS_SUBDIR_INHERIT;
if(strstr(ptr,"ACCESS_ALL")) lAccess=FTP_ACCESS_ALL;
p.second=lAccess;
}
ftpuser.dirAccess["/"]=p;
if( (it=maps.find("maxup"))!=maps.end())
{
ftpuser.maxupratio=atol((*it).second.c_str());
if(ftpuser.maxupratio<0) ftpuser.maxupratio=0;
}else ftpuser.maxupratio=0;
if( (it=maps.find("maxdw"))!=maps.end())
{
ftpuser.maxdwratio=atol((*it).second.c_str());
if(ftpuser.maxdwratio<0) ftpuser.maxdwratio=0;
}else ftpuser.maxdwratio=0;
if( (it=maps.find("maxupfile"))!=maps.end())
{
ftpuser.maxupfilesize=atol((*it).second.c_str());
if(ftpuser.maxupfilesize<0) ftpuser.maxupfilesize=0;
}else ftpuser.maxupfilesize=0;
if( (it=maps.find("maxlogin"))!=maps.end())
{
ftpuser.maxLoginusers=atol((*it).second.c_str());
if(ftpuser.maxLoginusers<0) ftpuser.maxLoginusers=0;
}else ftpuser.maxLoginusers=0;
if( (it=maps.find("expired"))!=maps.end())
{
structtm ltm; ::memset((void *)<m,0,sizeof(ltm));
::sscanf((*it).second.c_str(),"%d-%d-%d",<m.tm_year,<m.tm_mon,<m.tm_mday);
ltm.tm_year-=1900;//年份从1900开始计数
ltm.tm_mon-=1;//月份从0开始计数
if(ltm.tm_year>100 && ltm.tm_year<200 &&
ltm.tm_mon>=0 && ltm.tm_mon<=11 &&
ltm.tm_mday>=1 && ltm.tm_mday<=31 )
ftpuser.limitedTime=mktime(<m);
}else ftpuser.limitedTime=0;
if( (it=maps.find("maxdisksize"))!=maps.end())
{
long lsize=atol((*it).second.c_str());
ftpuser.maxdisksize=(lsize<0)?0:lsize;
constchar *ptr=strchr((*it).second.c_str(),':');
if(ptr) ftpuser.curdisksize=atol(ptr+1);
if(ftpuser.curdisksize<0) ftpuser.curdisksize=0;
}else ftpuser.maxdisksize=ftpuser.curdisksize=0;
if( (it=maps.find("hiddenfile"))!=maps.end())
ftpuser.disphidden=((*it).second=="HIDE" || (*it).second=="0")?0:1;
else ftpuser.disphidden=0;
if( (it=maps.find("pswdmode"))!=maps.end())
{
if((*it).second=="OTP_MD4")
ftpuser.pswdmode=OTP_MD4;
elseif((*it).second=="OTP_MD5")
ftpuser.pswdmode=OTP_MD5;
elseif((*it).second=="OTP_NONE")
ftpuser.pswdmode=OTP_NONE;
else ftpuser.pswdmode=atoi((*it).second.c_str());
}else ftpuser.pswdmode=0;
if( (it=maps.find("forbid"))!=maps.end())
{
if((*it).second=="true" || (*it).second=="1")
ftpuser.forbid=1;
else ftpuser.forbid=0;
}else ftpuser.forbid=0;
ftpuser.ipaccess=0;
ftpuser.ipRules="";
m_userlist[ftpuser.username]=ftpuser;
return ;
}
//设置ftp服务的ip过滤规则或针对某个帐号的IP过滤规则
//命令格式:
//iprules [account=<ftp帐号>] [access=0|1] ipaddr="<IP>,<IP>,..."
//account=<ftp帐号> : 指定此过滤规则是针对某个帐号的还是针对所有帐号适用的
//如果不设置此项则或ftp帐号设置为""或空则针对所有帐号适用
//access=0|1 : 对符合下列IP条件的是拒绝还是放行
//例如:
// iprules access=0 rules="192.168.0.*,192.168.1.10"
void ftpsvrEx :: docmd_iprules(constchar *strParam)
{
std::map<std::string,std::string> maps;
if(splitString(strParam,'',maps)<=0)return;
std::string strUser;
std::map<std::string,std::string>::iterator it;
if( (it=maps.find("account"))!=maps.end()) strUser=(*it).second;
std::map<std::string,TFTPUser>::iterator it1=m_userlist.find(strUser);
if(it1==m_userlist.end())return;
TFTPUser &ftpuser=(*it1).second;
if( (it=maps.find("access"))!=maps.end())
ftpuser.ipaccess=atoi((*it).second.c_str());
if( (it=maps.find("ipaddr"))!=maps.end())
ftpuser.ipRules=(*it).second;
return;
}
//添加修改删除ftp帐号的虚目录
//命令格式:
//vpath account=<ftp帐号> vdir=<虚目录> rdir=<真实目录> [access=<虚目录访问权限>]
//account=<ftp帐号> : 必须项. 要设置虚目录的ftp帐号。
//vdir=<虚目录> : 必须项. 要添加或修改的虚目录,每个虚目录必须以/开始,例如/aa
// 不能用此目录添加修改删除虚根目录。注意:虚目录区分大小写
//rdir=<真实目录> : 此虚目录对应的真实目录路径,必须是绝对路径,
// 如果<真实目录>中包含空格则要用""将<真实目录>括起,例如"c:\temp test\aa"
// 如果没有设置rdir或<真实目录>等于空字符串例如rdir= 或rdir=""则意味着删除此虚路径
//access=<虚目录访问权限> : 指定对此虚目录的访问权限。如果不设置则默认具有ACCESS_ALL访问权限。设置格式和含义如下
// <虚目录访问权限> : <FILE_READ|FILE_WRITE|FILE_DEL|FILE_EXEC|DIR_LIST|DIR_MAKE|DIR_DEL|DIR_NOINHERIT>
// ACCESS_ALL=FILE_READ|FILE_WRITE|FILE_DEL|FILE_EXEC|DIR_LIST|DIR_MAKE|DIR_DEL
// FILE_READ : 允许读文件 FILE_WRITE : 允许写文件 FILE_DEL : 允许删除文件 FILE_EXEC : 允许执行文件
// DIR_LIST : 允许目录文件list DIR_MAKE : 允许创建目录 DIR_DEL : 允许删除目录
// DIR_NOINHERIT : 是否允许此虚目录对应的真实路径下的子目录继承用户指定的目录访问权限。如果用户指定了此项则禁止继承,
// 那么默认ftp用户是无法对其子目录进行任何的读写操作,要想对其下的子目录具有操作,必须将其下的子目录映射为虚目录单独指定访问权限
// 注意:如果设定了FILE_EXEC ,则用户可通过扩展FTP命令EXEC,远程执行此虚目录下的文件。
void ftpsvrEx :: docmd_vpath(constchar *strParam)
{
std::map<std::string,std::string> maps;
if(splitString(strParam,'',maps)<=0)return;
std::map<std::string,std::string>::iterator it;
std::string strvdir;
if( (it=maps.find("account"))!=maps.end()) strvdir=(*it).second;
std::map<std::string,TFTPUser>::iterator it1=m_userlist.find(strvdir);
if(it1==m_userlist.end())return;
TFTPUser &ftpuser=(*it1).second;
std::pair<std::string,long>p("",0); strvdir="";
if( (it=maps.find("vdir"))!=maps.end()) strvdir=(*it).second;
if( strvdir=="")return;
if( (it=maps.find("rdir"))!=maps.end()) p.first=(*it).second;
if( (it=maps.find("access"))!=maps.end() && (*it).second!="")
{
constchar *ptr=(*it).second.c_str();
p.second=atol(ptr);
if(strstr(ptr,"FILE_READ")) p.second|=FTP_ACCESS_FILE_READ;
if(strstr(ptr,"FILE_WRITE")) p.second|=FTP_ACCESS_FILE_WRITE;
if(strstr(ptr,"FILE_DEL")) p.second|=FTP_ACCESS_FILE_DELETE;
if(strstr(ptr,"FILE_EXEC")) p.second|=FTP_ACCESS_FILE_EXEC;
if(strstr(ptr,"DIR_LIST")) p.second|=FTP_ACCESS_DIR_LIST;
if(strstr(ptr,"DIR_MAKE")) p.second|=FTP_ACCESS_DIR_CREATE;
if(strstr(ptr,"DIR_DEL")) p.second|=FTP_ACCESS_DIR_DELETE;
if(strstr(ptr,"DIR_NOINHERIT")) p.second|=FTP_ACCESS_SUBDIR_INHERIT;
if(strstr(ptr,"ACCESS_ALL")) p.second=FTP_ACCESS_ALL;
}
ftpuser.dirAccess[strvdir]=p;
return;
}