\n",\ vsurfdata[ii].obtid,vsurfdata[ii].ddatetime,vsurfdata[ii].t/10.0,vsurfdata[ii].p/10.0,\ vsurfdata[ii].u,vsurfdata[ii].wd,vsurfdata[ii].wf/10.0,vsurfdata[ii].r/10.0,vsurfdata[ii].vis/10.0); } else { File.Fprintf("%s,%s,%.1f,%.1f,%d,%d,%.1f,%.1f,%.1f\n",\ vsurfdata[ii].obtid,vsurfdata[ii].ddatetime,vsurfdata[ii].t/10.0,vsurfdata[ii].p/10.0,\ vsurfdata[ii].u,vsurfdata[ii].wd,vsurfdata[ii].wf/10.0,vsurfdata[ii].r/10.0,vsurfdata[ii].vis/10.0); } } if (strcmp(datafmt,"xml")==0) File.Fprintf("\n"); //结尾加 File.CloseAndRename(); // 关闭文件 logfile.Write("生成数据文件(%s)成功,数据时间=%s,记录数=%d!\n\n",strFileName,vsurfdata[0].ddatetime,vsurfdata.size()); vstcode.clear(); vsurfdata.clear(); return true; } void EXIT(int sig) { logfile.Write("程序退出,sig=%d\n\n",sig); exit(0); }
如下生成一个xml文件里也有800多行,最后也以结尾
如下是主动模式
如下是被动模式(经常用),设置ftp高端端口即数据端口
Linux操作系统的用户也是ftp的用户,可以配置专用的ftp用户,专用的ftp用户只能用于ftp,不能登录操作系统。如下是ftp安装:
安装ftp客户端
:yum -y install ftp (remove)(只要ftp服务端安装并配置好防火墙等,任何主机安装了客户端都可以连上)。
安装ftp服务端
:yum -y install vsftpd (本地和远程服务器都可安装)。如下是ftp服务端配置:
1.
防火墙开启 21 端口和5500-6000端口
如果采用被动模式
,防火墙开通21端口
(命令含义: --zone#作用域 --add-port=21/tcp#添加端口,格式为:端口/通讯协议 --permanent # 永久生效)
#firewall-cmd --zone=public --add-port=21/tcp --permanent
如果采用主动模式
,防火墙还要开通20端口
#firewall-cmd --zone=public --add-port=20/tcp --permanent
#firewall-cmd --zone=public --add-port=5500-6000/tcp --permanent
(设置被动模式的高端口范围为5500-6000,passive命令为on,完成后必须init 6重启,netstat -na --ip,firewall-cmd --list-ports,防火墙配置不好可以直接关闭防火墙)
2.
配置ftp高端口参数和关闭 selinux
#vi /etc/vsftpd/vsftpd.conf
(必选装好vsftpd服务端,init6)
listen=YES //将上面含有listen都注释掉
listen_ipv6=NO //关闭ipv6的监听,不然会ls,dir不出命令
pam_service_name=vsftpd
userlist_enable=YES
tcp_wrappers=YES
pasv_min_port=5500
pasv_max_port=6000
pasv_enable=YES
pasv_address=47.100.167.156 //服务端公网ip
#vi /etc/selinux/config
,修改成 selinux=disabled,执行 setenforce 0 使修改马上生效
3.
登录阿里云控制台实例安全组(如上配置不成功,手动配置端口)
点击安全组列表中配置规则
4.
vsftpd服务端相关操作
systemctl start vsftpd # 启动
systemctl stop vsftpd # 停止
systemctl status vsftpd # 查看
systemctl enable vsftpd # 开机自动启动vsftpd服务(类似windows下服务)
systemctl disable vsftpd # 禁用vsftpd服务
5.
防火墙服务相关操作
以下是centos7
的命令:
centos7中的防火墙名改成了firewall
systemctl restart firewalld.service # 重启防火墙
systemctl stop firewalld.service # 关闭防火墙
systemctl start firewalld.service # 启动防火墙
systemctl status firewalld.service # 查看防火墙服务状态
systemctl enable firewalld.service # 开机禁用防火墙
以下是centos6
的命令:
chkconfig iptables off # 开机禁用防火墙
service vsftpd restart # 重启ftp服务端
service iptables stop #关闭防火墙
方法一
:输入ftp ip地址,然后输入用户名和密码(adduser ,passwd )
方法二
:输入ftp,用open连上服务器,再输入用户名和密码,如下图:
方法三
:输入ftp -n ip地址,用user命令登录,如下图:进入ftp服务后输入open加ip地址open 118.89.50.198
dir
:显示服务器目录和文件列表
ls
:显示服务器目录和文件列表
cd
:进入服务器指定的目录
lcd
:进入本地客户端指定的目录。lcd空可查看本地位置
(dir命令可以使用通配符“*”和“?”,比如,显示当前目录中所有扩展名为jpg的文件,可使用命令 dir *.jpg。cd命令中必须带目录名。比如 cd /tmp 表示进入/tmp目录)
ftp的传输模式分为ASCII码方式和二进制方式两种,二进制方式可以传输任何文件,包括压缩包、可执行程序、图片、视频、音频等,而ASCII码方式只能传输.txt、.htm等ascii码文件。在实际开发中,不管什么文件都用二进制方式传输文件。
type
:查看当前的传输方式
ascii
:设定传输方式为ASCII码方式
binary
:设定传输方式为二进制方式
get/recv
:下载单个文件get filename [newname]
(filename为下载的ftp服务器上的文件名,newname为保存在本都计算机上时使用的名字,get/recv只能取文件(文件夹需要打包压缩),加-r
可以操作文件夹
mget
:下载多个文件mget filename [filename ....]
(mget命令支持通配符“*”和“?”,比如:mget *.jpg 表示下载ftp服务器当前目录下的所有扩展名为jpg的文件。)
prompt
:关闭/打开互交提示。关闭后,mget不用输入y
put/send
:上传单个文件put filename [newname],filename为上传的本地文件名,newname为上传至ftp服务器上时使用的名字,如果不指定newname,文件将以原名上传。
mput
:上传多个文件mput filename [filename …],mput命令支持通配符“*”和“?”
bye
:结束与服务器的ftp会话并退出ftp环境
pwd
:查看ftp服务器上的当前工作目录
rename filename newfilename
:重命名ftp服务器上的文件
delete filename
:删除ftp服务器上一个文件。
mdelete [remote-files]
:删除多个文件。
mkdir pathname
:在服务器上创建目录。
rmdir pathname
:删除服务器上的目录。
passive
:主动模式与被动模式切换。
nlist
:列出服务器目录中的文件名,如:nlist /home/w /tmp/tmp.list,表示把服务器上/home/w目录下的文件列出来,结果输出到本地的/tmp/tmp.list文件中,输出文件名是全路径。
help [cmd]
:显示ftp命令的帮助信息,cmd是命令名,如果不带参数,则显示所有ftp命令。
windows的命令提示符下有ftp客户端程序,但是不好用。 采用资源管理器,输入:ftp://服务器ip,如下图:
在空白的位置点鼠标右键,选择登录菜单, 输入用户名和密码登录ftp服务器界面,直接将文件拖到windows界面
vi ftp.sh,mdelete* 将服务器上文件删掉,共享服务器不能删,取的时候有文件生成,不知道哪些文件取过或哪些没取过,所以这种方法不可行
在一文件夹里:ls | wc 统计数据行数。服务器上一生成文件,本地虚拟机就取回来。以下用C语言实现ftp客户端,File.Open>写文件>File.close
是需要时间的,写文件过程中ftp将这文件取走的话,取走的文件不完整,如果是银行数据不可行。如下有两个数据不完整不安全环节
第一环节供ftp用:
创建文件SNPRINTF命名为.txt.tmp,读取Open文件.txt.tmp,数据写完后才会close再rename为.txt,避免文件还没Close被取走
第二环节供处理数据程序用:
避免ftp没有get完被处理数据程序取走
下面在本地rename
以上就能避免中间状态文件被读取,下面修改crtsurfdata.cpp
sleep(20);// 停20秒方便观看
下面把上面封装起来,Open和Close时候就分别命名,写入File这个类中
上面是解决临时文件问题,下面是C语言实现ftp下载文件。如下demo18用到_ftp.h(连着_ftp.cpp一起编译),_ftp.h又用到ftplib.h(连着ftplib.c一起编译,.h必用到其实现的.c/.cpp) 。
.a就是静态库相当.c/.cpp一个别名,编译时不让别人看见,makefile中libftp.a就是ftplib.c
。ftplib.h和ftplib.c
源代码链接:https://pan.baidu.com/s/1P_GTtiOpMnO3KqU6_VVnzQ 提取码:u77a
CC=g++
FLAG=-g
#FLAG=-02
all:crtsurfdata libftp.a demo18 ftpgetfiles
libftp.a:ftplib.c
gcc -c -o libftp.a ftplib.c
demo18:demo18.cpp _ftp.h _ftp.cpp libftp.a
g++ $(FLAGS) -o demo18 demo18.cpp _public.cpp _ftp.cpp libftp.a
ftpgetfiles:ftpgetfiles.cpp _ftp.h _ftp.cpp libftp.a
$(CC) $(FLAG) -o ftpgetfiles ftpgetfiles.cpp _public.cpp _ftp.cpp libftp.a
cp ftpgetfiles ../bin/.
crtsurfdata:crtsurfdata.cpp _public.h _public.cpp
$(CC) $(FLAG) -o crtsurfdata crtsurfdata.cpp _public.cpp
cp crtsurfdata ../bin/.
clean:
rm -rf crtsurfdata demo18 ftpgetfiles libftp
// 本程序演示Cftp客户端类,采用ftp协议从服务器上获取文件,demo18
#include "_ftp.h"
int main(int argc,char *argv[])
{
Cftp ftp;
// 登录远程FTP服务器
if (ftp.login("193.112.16.23:21","用户名","密码",FTPLIB_PASSIVE) == false)
{
printf("ftp.login() failed.\n"); return -1;
}
ftp.mtime("/home/y/surfdata/SURF_ZH_20200209122402_2815.txt");
ftp.size("/home/y/surfdata/SURF_ZH_20200209122402_2815.txt");
printf("=%s=\n",ftp.response());
// 改变本地目录
chdir("/tmp");
// 进入ftp服务器上文件存放的目录
if (ftp.chdir("/home/y/surfdata") == false)
{
printf("ftp.chdir() failed.\n"); return -1;
}
// 获取对方目录文件的列表,存放在"/tmp/ftp.list"文件中
if (ftp.nlist("*.txt","/tmp/ftp.list")== false)
{
printf("ftp.nlist() failed.\n"); return -1;
}
CFile File;
File.OpenForRead("/tmp/ftp.list","r");
// 逐行读取文件的内容,并把文件get到本地
char strFileName[101];
while (true)
{
// 从文件中读取一行
memset(strFileName,0,sizeof(strFileName));
if (File.Fgets(strFileName,100) == false) break;
strFileName[strlen(strFileName)-1]=0; //去除最后回车符
printf("get %s ...",strFileName);
// 从远程取文件
if (ftp.get(strFileName,strFileName)==false)
{
printf("ftp.get(%s) failed.\n",strFileName); break;
}
printf("ok.\n");
}
File.CloseAndRemove();
ftp.logout();
return 0;
}
dir和nlist差不多,dir相当于ls -l。判断文件内容是否改变判断文件时间
,不能判断大小
// _ftp.h
#ifndef __FTP_H
#define __FTP_H
#include "_public.h"
#include "ftplib.h"
class Cftp
{
public:
// ftp连接句柄
netbuf *m_ftpconn;
// 文件的大小
unsigned int m_size;
// 文件的时间modifytime
char m_mtime[21];
Cftp();
~Cftp();
// 存放login()方法登录失败的原因
bool m_connectfailed;
bool m_loginfailed;
bool m_optionfailed;
void initdata();
// 登录ftp服务器
// in_host 服务器地址和端口,中间用":"分隔,如"192.168.1.1:21"
// in_username ftp用户名
// in_password ftp的密码
// in_mode 传输模式,FTPLIB_PASSIVE是被动模式,FTPLIB_PORT是主动模式
bool login(const char *in_host,const char *in_username,const char *in_password,const int in_mode=FTPLIB_PASSIVE);
// 注销
bool logout();
// 获取ftp服务器上文件的时间
bool mtime(const char *in_remotefilename);
// 获取ftp服务器上文件的大小
bool size(const char *in_remotefilename);
// 向服务端发送site命令
bool site(const char *in_command);
// 改变ftp远程目录
bool chdir(const char *in_remotedir);
// 在ftp服务器上创建目录
bool mkdir(const char *in_remotedir);
// 删除ftp服务器上的目录
bool rmdir(const char *in_remotedir);
// 发送list命令列出ftp服务器目录中的文件,结果保存到本地文件中
// 如果是列出当前目录,in_remotedir用"","*","."都行。
bool nlist(const char *in_remotedir,const char *out_listfilename);
// 发送dir命令列出ftp服务器目录中的文件,结果保存到本地文件中
bool dir(const char *in_remotedir,const char *out_listfilename);
// 从ftp服务器上获取文件
// in_remotefilename 待获取的远程文件名
// in_localfilename 本地文件名,可以与in_remotefilename不同
// bCheckMTime 文件传输完成后,是否核对文件传输前后的时间,保证文件的完整性
// 注意,文件在传输的过程中,采用临时文件命名的方法,即在in_localfilename后加".tmp",在传输
// 完成后才正式改为in_localfilename
bool get(const char *in_remotefilename,const char *in_localfilename,const bool bCheckMTime=true);
// 向ftp服务器发送文件
// in_localfilename 本地待发送的文件名
// in_remotefilename 远程文件名
// bCheckSize 文件传输完成后,是否核对本地和远程文件的大小,保证文件的完整性
// 注意,文件在传输的过程中,采用临时文件命名的方法,即在in_remotefilename后加".tmp",在传输
// 完成后才正式改为in_remotefilename
bool put(const char *in_localfilename,const char *in_remotefilename,const bool bCheckSize=true);
// 删除ftp服务器上的文件
bool ftpdelete(const char *in_remotefilename);
// 把ftp服务器上的文件改名
bool ftprename(const char *in_srcremotefilename,const char *in_dstremotefilename);
// 获取服务器返回信息的最后一条 return a pointer to the last response received
char *response();
};
#endif
如下bool Cftp::mtime()中AddTime()在_public.cpp实现,注意"yyyymmddhh24miss"格式
// _ftp.cpp
#include "_ftp.h"
Cftp::Cftp()
{
m_ftpconn=0;
initdata();
FtpInit();
m_connectfailed=false;
m_loginfailed=false;
m_optionfailed=false;
}
Cftp::~Cftp()
{
logout();
}
void Cftp::initdata()
{
m_size=0;
memset(m_mtime,0,sizeof(m_mtime));
}
bool Cftp::login(const char *in_host,const char *in_username,const char *in_password,const int in_mode)
{
if (m_ftpconn != 0) { FtpQuit(m_ftpconn); m_ftpconn=0; }
m_connectfailed=m_loginfailed=m_optionfailed=false;
if (FtpConnect(in_host,&m_ftpconn) == false) { m_connectfailed=true; return false; }
if (FtpLogin(in_username,in_password,m_ftpconn) == false) { m_loginfailed=true; return false; }
if (FtpOptions(FTPLIB_CONNMODE,(long)in_mode,m_ftpconn) == false) { m_optionfailed=true; return false; }
return true;
}
bool Cftp::logout()
{
if (m_ftpconn == 0) return false;
FtpQuit(m_ftpconn);
m_ftpconn=0;
return true;
}
bool Cftp::get(const char *in_remotefilename,const char *in_localfilename,const bool bCheckMTime)
{
if (m_ftpconn == 0) return false;
// 创建本地文件目录
MKDIR(in_localfilename);
char strlocalfilenametmp[301];
memset(strlocalfilenametmp,0,sizeof(strlocalfilenametmp));
snprintf(strlocalfilenametmp,300,"%s.tmp",in_localfilename);
// 获取远程服务器的文件的时间
if (mtime(in_remotefilename) == false) return false;
// 取文件
if (FtpGet(strlocalfilenametmp,in_remotefilename,FTPLIB_IMAGE,m_ftpconn) == false) return false;
// 判断文件获取前和获取后的时间,如果时间不同,表示文件已改变,返回失败
if (bCheckMTime==false)
{
char strmtime[21];
strcpy(strmtime,m_mtime);
if (mtime(in_remotefilename) == false) return false;
if (strcmp(m_mtime,strmtime) != 0) return false;
}
// 重置文件时间
UTime(strlocalfilenametmp,m_mtime);
// 改为正式的文件
if (rename(strlocalfilenametmp,in_localfilename) != 0) return false;
// 获取文件的大小
m_size=FileSize(in_localfilename);
return true;
}
bool Cftp::mtime(const char *in_remotefilename)
{
if (m_ftpconn == 0) return false;
memset(m_mtime,0,sizeof(m_mtime));
char strmtime[21];
memset(strmtime,0,sizeof(strmtime));
if (FtpModDate(in_remotefilename,strmtime,14,m_ftpconn) == false) return false;
AddTime(strmtime,m_mtime,0+8*60*60,"yyyymmddhh24miss");
return true;
}
bool Cftp::size(const char *in_remotefilename)
{
if (m_ftpconn == 0) return false;
m_size=0;
if (FtpSize(in_remotefilename,&m_size,FTPLIB_IMAGE,m_ftpconn) == false) return false;
return true;
}
bool Cftp::site(const char *in_command)
{
if (m_ftpconn == 0) return false;
if (FtpSite(in_command,m_ftpconn) == false) return false;
return true;
}
bool Cftp::chdir(const char *in_remotedir)
{
if (m_ftpconn == 0) return false;
if (FtpChdir(in_remotedir,m_ftpconn) == false) return false;
return true;
}
bool Cftp::mkdir(const char *in_remotedir)
{
if (m_ftpconn == 0) return false;
if (FtpMkdir(in_remotedir,m_ftpconn) == false) return false;
return true;
}
bool Cftp::rmdir(const char *in_remotedir)
{
if (m_ftpconn == 0) return false;
if (FtpRmdir(in_remotedir,m_ftpconn) == false) return false;
return true;
}
bool Cftp::dir(const char *in_remotedir,const char *out_listfilename)
{
if (m_ftpconn == 0) return false;
if (FtpDir(out_listfilename,in_remotedir,m_ftpconn) == false) return false;
return true;
}
bool Cftp::nlist(const char *in_remotedir,const char *out_listfilename)
{
if (m_ftpconn == 0) return false;
// 创建本地文件目录
MKDIR(out_listfilename);
if (FtpNlst(out_listfilename,in_remotedir,m_ftpconn) == false) return false;
return true;
}
bool Cftp::put(const char *in_localfilename,const char *in_remotefilename,const bool bCheckSize)
{
if (m_ftpconn == 0) return false;
char strremotefilenametmp[301];
memset(strremotefilenametmp,0,sizeof(strremotefilenametmp));
snprintf(strremotefilenametmp,300,"%s.tmp",in_remotefilename);
if (FtpPut(in_localfilename,strremotefilenametmp,FTPLIB_IMAGE,m_ftpconn) == false) return false;
if (FtpRename(strremotefilenametmp,in_remotefilename,m_ftpconn) == false) return false;
// 判断已上传的文件的大小与本地文件是否相同,确保上传成功。
if (bCheckSize==true)
{
if (size(in_remotefilename) == false) return false;
if (m_size != FileSize(in_localfilename)) return false;
}
return true;
}
bool Cftp::ftpdelete(const char *in_remotefilename)
{
if (m_ftpconn == 0) return false;
if (FtpDelete(in_remotefilename,m_ftpconn) == false) return false;
return true;
}
bool Cftp::ftprename(const char *in_srcremotefilename,const char *in_dstremotefilename)
{
if (m_ftpconn == 0) return false;
if (FtpRename(in_srcremotefilename,in_dstremotefilename,m_ftpconn) == false) return false;
return true;
}
char *Cftp::response()
{
if (m_ftpconn == 0) return 0;
return FtpLastResponse(m_ftpconn);
}
结构化数据:
有记录有条数,操作数据库,传文件,调http接口
非结构化数据:
图片,视频,word文档,ftp
(数据中心中结构化数据用操作数据库推上云,非结构化数据用ftp推上云)
下面在start.sh中
上面strbuffer[…-1]只删除linux下\n换行符效果如下,window下\r不能删除,也可能会有两个\r
// 从ftp服务器上采集文件,ftpgetfiles.cpp
#include"_public.h"
#include"_ftp.h"
struct st_arg
{
char host[51];
int mode;
char username[31];
char password[31];
char localpath[301];
char remotepath[301];
char matchname[301];
int ptype;
char remotepathbak[301];
char listfilename[301];
char okfilename[301];
int timetvl;
} starg;
/*
struct st_fileinfo
{ //在_public.h中定义了
char filename[301];
char mtime[21];
};
*/
vector vokfilename,vokfilename1;
vector vlistfile,vlistfile1;
Cftp ftp;
CLogFile logfile; //全局变量,日志操作类对象
//把nlist获取的文件名加载到vlistfile容器中
bool LoadListFile();
//把okfilename文件的内容加载到voklistname容器中
bool LoadOKFileName();
//比较vlistfile容器与vokfilename中的文件,得到新的两个容器
bool CompVector();
//把vokfilename1容器里的内容先写入okfilename文件中,覆盖之前的旧okfilename文件
bool WriteToOKFileName();
//ptype=1,把采集成功的文件追加到okfilename文件中
bool AppendToOKFileName(struct st_fileinfo *stfileinfo);
//退出信号的处理函数
void EXIT(int sig);
//本程序的业务流程的主函数
bool _ftpgetfile();
//显示文件的帮助
void _help(char *argv[]);
//把xml参数读取到starg结构体中
bool _xmltoarg(char *strxmlbuffer);
int main(int argc,char *argv[])
{
if(argc != 3)
{
_help(argv); //argv是数组名也就是地址
return -1;
}
//关闭全部的信号和输入输出
CloseIOAndSignal();
//处理程序退出的信号
signal(SIGINT,EXIT); //中断信号
signal(SIGTERM,EXIT); //kill信号
if(logfile.Open(argv[1],"a+") == false)
{
logfile.Write("打开日志文件(%s)失败。\n",argv[1]); // logout会自动调用_ftp.h里析构函数,所以不用写
return -1;
}
if(_xmltoarg(argv[2]) == false)
return -1;
while(true)
{
if((ftp.login(starg.host,starg.username,starg.password,starg.mode)) == false)
{
logfile.Write("ftp.login(%s,%s,%s) failed.\n",starg.host,starg.username,starg.password);
//return -1;
sleep(starg.timetvl);
continue;
}
//logfile.Write("ftp.login ok.\n");
//本程序的业务流程的主函数
_ftpgetfile();
ftp.logout();
sleep(starg.timetvl);
}
return 0;
}
//////////////////////////////////////////1.本程序的业务流程的主函数
bool _ftpgetfile()
{
//先chdir进入目录,这样listfilename输出就不是全路径了,节省文件大小,减小网络开销
if(ftp.chdir(starg.remotepath) == false)
{
logfile.Write("ftp.chdir(%s) failed.\n",starg.remotepath);
return false;
}
//logfile.Write("chdir ok.\n");
//列出服务器目录文件 //每次取时和listfilename做对比实现增量采集
if(ftp.nlist(".",starg.listfilename) == false)
{
logfile.Write("ftp.nlist(%s) failed.\n",starg.remotepath);
return false;
}
//logfile.Write("nlist ok.\n");
//把nlist获取的文件加载到vlistfile容器中,一般LoadListFile不会返回失败,但是里面有ftp.mtime联网可能会返回失败
if(LoadListFile() == false)
{
logfile.Write("LoadListFile failed.\n");
return -1;
}
//chdir(starg.localpath); //切换本地工作目录,如果这目录不存在就切换不成功,所以不用这方法,用绝对路径
if(starg.ptype == 1)
{
//okfilename是xml文件格式SURF_ZH_20190913131401_22226.txt20191026212900
//加载okfilename文件里的内容到容器vokfilename里
LoadOKFileName();
//把vlistfile容器中的文件与vokfilename中的文件对比,得到新的两个容器
//1.在vlistfile中存在,并已经采集成功的文件vokfilename1
//2.在vlistfile中存在,新文件(vlistfile有,vokfilename没有)或要重新采集(vlistfile与vokfilename文件时间不同)的文件vlistfile1
CompVector();
//把vokfilename1容器里的内容先写入okfilename文件中,覆盖之前的旧okfilename文件
WriteToOKFileName();
//把vlistfile1容器中的内容复制到vlistfile容器中让下面代码都可用
//vlistfile.clear();
vlistfile.swap(vlistfile1);
}
for(int i=0;i<vlistfile.size();i++)
{
char strremotefilename[301],strlocalfilename[301]; //不切换本地工作目录了,定义两个变量拼成绝对路径
SNPRINTF(strlocalfilename,300,"%s/%s",starg.localpath,vlistfile[i].filename);
SNPRINTF(strremotefilename,300,"%s/%s",starg.remotepath,vlistfile[i].filename);
//vlistfile为string时,vlistfile[i].c_str()。为struct时push_back进去,vlistfile[i].filename
logfile.Write("get %s ...",strremotefilename);
//获取文件
if(ftp.get(strremotefilename,strlocalfilename,true) == false )
{
logfile.WriteEx("failed.\n"); //WriteEx不写时间
break;
}
logfile.WriteEx("ok.\n");
//删除文件
if (starg.ptype==2) ftp.ftpdelete(strremotefilename);
//转存到备份目录
if(starg.ptype == 3)
{
char strremotefilenamebak[301]; //remotepathbak不要自己创建
SNPRINTF(strremotefilenamebak,300,"%s/%s",starg.remotepathbak,vlistfile[i].filename); //拼成
ftp.ftprename(strremotefilename,strremotefilenamebak);
}
//ptype=1,把采集成功的文件追加到okfilename文件中
if(starg.ptype == 1)
{
AppendToOKFileName(&vlistfile[i]);
}
}
return true;
}
void EXIT(int sig)
{
logfile.Write("程序退出,sig=%d\n",sig);
exit(0);
}
///////////////////////////////////////////////2.加载对方服务器文件列表
bool LoadListFile() // 和crtsurfdata.cpp中LoadSTCode()差不多
{
vlistfile.clear();
CFile file;
if(file.Open(starg.listfilename,"r") == false)
{
logfile.Write("file.Open(%s) failed.\n",starg.listfilename);
return false;
}
struct st_fileinfo stfileinfo;
while(true)
{
memset(&stfileinfo,0,sizeof(stfileinfo));
if(file.Fgets(stfileinfo.filename,300,true) == false)
break;
//或许放在对比完下载文件的时候更合适?
if(MatchFileName(stfileinfo.filename,starg.matchname) == false)
continue;
if(starg.ptype == 1 )
{
//获取服务器文件时间 //这边可能会网络交互时返回失败,所以记日志 //只有ptype == 1才需要获取文件时间
if(ftp.mtime(stfileinfo.filename) == false)
{
logfile.Write("ftp.mtime(%s) failed.\n",stfileinfo.filename);
return false;
}
strcpy(stfileinfo.mtime,ftp.m_mtime); //m_mtime在_ftp.h中AddTime,mtime成员函数中
}
vlistfile.push_back(stfileinfo);
//logfile.Write("vlistfile filename=%s,mtime=%s\n",stfileinfo.filename,stfileinfo.mtime);
}
return true;
}
////////////////////////////////////////////3.加载采集成功文件列表
bool LoadOKFileName()
{
vokfilename.clear();
CFile file;
//如果程序是第一次采集,okfilename是不存在的,并不是错误,也返回true
if(file.Open(starg.okfilename,"r") == false)
return true;
struct st_fileinfo stfileinfo;
char strbuffer[301];
while(true)
{
memset(&stfileinfo,0,sizeof(stfileinfo));
if(file.Fgets(strbuffer,300,true) == false)
break;
GetXMLBuffer(strbuffer,"filename",stfileinfo.filename,300);
GetXMLBuffer(strbuffer,"mtime",stfileinfo.mtime,20);
vokfilename.push_back(stfileinfo);
//logfile.Write("vokfilename filename=%s,mtime=%s\n",stfileinfo.filename,stfileinfo.mtime);
}
return true;
}
//////////////////4.把vlistfile容器中的文件与vokfilename中的文件对比,得到新的两个容器
bool CompVector()
{
vokfilename1.clear();
vlistfile1.clear();
for(int i=0;i<vlistfile.size();i++) //这个循环得到vlistfile1
{
int j=0; //在外面定义下面可以用j
for(j=0;j<vokfilename.size();j++) //这个循环得到vokfilename1
{
if( (strcmp(vlistfile[i].filename,vokfilename[j].filename)) == 0 &&
(strcmp(vlistfile[i].mtime,vokfilename[j].mtime)) == 0 ) //这两个都等的话不需要再取
{
vokfilename1.push_back(vlistfile[i]); break; //不满足上面两个条件,break跳出
}
}
if(j == vokfilename.size()) //因为j<vokfilename.size(),所以这行意思是肯定没找到两个相等的
vlistfile1.push_back(vlistfile[i]); //if省略了{}
}
/*
for(int i=0;i<vlistfile1.size();i++)
{
logfile.Write("vlistfile1 filename=%s,mtime=%s\n",vlistfile1[i].filename,vlistfile1[i].mtime);
}
for(int i=0;i<vokfilename1.size();i++)
{
logfile.Write("vokfilename1 filename=%s,mtime=%s\n",vokfilename1[i].filename,vokfilename1[i].mtime);
}
*/
return true;
}
///////////////////5.把vokfilename1容器里的内容先写入okfilename文件中,覆盖之前的旧okfilename文件
bool WriteToOKFileName()
{
CFile file;
if(file.Open(starg.okfilename,"w") == false)
{
logfile.Write("file.Open(%s) failed.\n",starg.okfilename);
return false;
}
for(int i=0;i<vokfilename1.size();i++)
{
file.Fprintf("%s%s\n",vokfilename1[i].filename,vokfilename1[i].mtime);
}
return true;
}
/////////////////////////6.ptype=1,把采集成功的文件追加到okfilename文件中
bool AppendToOKFileName(struct st_fileinfo *stfileinfo)
{
CFile file;
if(file.Open(starg.okfilename,"a") == false)
{
logfile.Write("file.Open(%s) failed.\n",starg.okfilename);
return false;
}
file.Fprintf("%s%s\n",stfileinfo->filename,stfileinfo->mtime);
return true;
}
///////////////////////////////////////7.把xml参数读取到starg结构体中
bool _xmltoarg(char *strxmlbuffer)
{
memset(&starg,0,sizeof(struct st_arg));
GetXMLBuffer(strxmlbuffer,"host",starg.host);
if(strlen(starg.host) == 0)
{ logfile.Write("host is null.\n"); return false;}
GetXMLBuffer(strxmlbuffer,"mode",&starg.mode);
if( (starg.mode != 1) && (starg.mode != 2) )
starg.mode = 1;
GetXMLBuffer(strxmlbuffer,"username",starg.username);
if(strlen(starg.username) == 0)
{ logfile.Write("username is null.\n"); return false;}
GetXMLBuffer(strxmlbuffer,"password",starg.password);
if(strlen(starg.password) == 0)
{ logfile.Write("password is null.\n"); return false;}
GetXMLBuffer(strxmlbuffer,"localpath",starg.localpath);
if(strlen(starg.localpath) == 0)
{ logfile.Write("localpath is null.\n"); return false;}
GetXMLBuffer(strxmlbuffer,"remotepath",starg.remotepath);
if(strlen(starg.remotepath) == 0)
{ logfile.Write("remotepath is null.\n"); return false;}
GetXMLBuffer(strxmlbuffer,"matchname",starg.matchname);
if(strlen(starg.matchname) == 0)
{ logfile.Write("matchname is null.\n"); return false;}
GetXMLBuffer(strxmlbuffer,"ptype",&starg.ptype);
if( (starg.ptype != 1) && (starg.ptype != 2) && (starg.ptype != 3))
{ logfile.Write("ptype is error.\n"); return false;}
GetXMLBuffer(strxmlbuffer,"remotepathbak",starg.remotepathbak);
if( (starg.ptype == 3) && (strlen(starg.remotepathbak) == 0))
{ logfile.Write("remotepathbak is null.\n"); return false;}
GetXMLBuffer(strxmlbuffer,"listfilename",starg.listfilename);
if(strlen(starg.listfilename) == 0)
{ logfile.Write("listfilename is null.\n"); return false;}
GetXMLBuffer(strxmlbuffer,"okfilename",starg.okfilename);
if( (starg.ptype == 1) && (strlen(starg.okfilename) == 0))
{ logfile.Write("okfilename is null.\n"); return false;}
GetXMLBuffer(strxmlbuffer,"timetvl",&starg.timetvl);
if( (starg.timetvl) == 0)
{ logfile.Write("timetvl is null.\n"); return false;}
return true;
}
//显示文件的帮助
void _help(char *argv[])
{
printf("\n公共模块,从ftp服务器上采集文件。\n");
printf("/root/qxidc/bin/ftpgetfile logfilename xmlbuffer \n\n");
printf("例如:/root/qxidc/bin/ftpgetfile /root/qxidc/log/ftpgetfile.log \"47.100.16.15:211y5199/root/qxidc/data/surfdata/home/yu/surfdataSURF_*.txt,*.DAT1/root/qxidc/ftplist/ftpgetfile_surfdata.list/root/qxidc/ftplist/ftpgetfile_surfdata.xml30 \"\n\n");
printf("logfilename 程序运行的日志文件名 \n");
printf("xmlbuffer 文件传输的参数,如下: \n");
printf("118.31.74.14:21 远程服务器的IP和端口 \n");
printf("1 传输模式,1-pasv(被动),2-port(主动),缺省为1 \n");
printf("gu 远程服务器的ftp用户名 \n");
printf("98 远程服务器的ftp密码 \n");
printf("/home/gu/tmp/ftpget 本地文件存放的目录 \n");
printf("/home/y/ 远程服务器文件存放的目录 \n");
printf("*.GIF 待采集文件匹配的文件名,大写匹配,"\
"不匹配的文件不会被采集,尽量精确,不允许采用*匹配全部文件\n");
printf("2 采集之后,服务器处理方式:1-什么都不做(增量采集),2-删除,3-备份 \n");
printf("/home/y/bak 远程服务器文件备份的目录,只有当ptype=3时才有效 \n");
printf("/home/gu/qxidc/list/ftpgetfile_surfdata.list 采集器列出服务器文件名的清单 \n");
printf("/home/gu/qxidc/list/ftpgetfile_surfdata.xml 已采集成功文件名清单 \n");
printf("30 采集时间间隔 单位:s 一般大于 10 \n\n");
//不能多个程序共用listfilename和okfilename
}