上次中实现了FTP命令的映射来避免很多if....else的判断,这次主要是开始实现目录列表的传输,先看一下目前实现的:

数据连接创建好之后则开始进行目录列表的传输了,而要传输目录列表,首先要将目录列表这些数据获取出来,所以这次主要是完成目录列表数据的获取。

还是对照着vsftpd的目录列表的显示来做参考:

将其列表输出拷贝出来:

其它跟"ls -l"的输出形式类似:

下面定义一个函数来将当前目录列出来:

int list_common(void)
{
//打开当前目录
DIR *dir = opendir(".");
if (dir == NULL)
{
return ;
} return ;
}

接下来读取目录进行遍历,可以通过如下函数:

【说明】:关于文件目录操作的知识,在之前的系统编程中已经学过了。

int list_common(void)
{
//打开当前目录
DIR *dir = opendir(".");
if (dir == NULL)
{
return ;
} //读取目录并进行遍历
struct dirent *dt;
while ((dt = readdir(dir)) != NULL)
{
} return ;
}

接下来读取文件的状态,并按vsftpd的输出格式来输出:

首先需要获取文件的状态,可以通过如下函数:

上面有两个获取文件状态的函数:stat、lstat,那这里要使用哪一个呢?这里要使用lstat,先看一下他们两者的区别:

所以代码如下:

int list_common(void)
{
//打开当前目录
DIR *dir = opendir(".");
if (dir == NULL)
{
return ;
} //读取目录并进行遍历
struct dirent *dt;
struct stat sbuf;
while ((dt = readdir(dir)) != NULL)
{ //获取文件的状态
if (lstat(dt->d_name, &sbuf) < 0)
{
continue;
}
} return ;
}

接下来从文件状态中来读取权限位,首先读取的是文件类型:

而它的获取,可以通过它来:

具体代码如下:

int list_common(void)
{
//打开当前目录
DIR *dir = opendir(".");
if (dir == NULL)
{
return ;
} //读取目录并进行遍历
struct dirent *dt;
struct stat sbuf;
while ((dt = readdir(dir)) != NULL)
{ //获取文件的状态
if (lstat(dt->d_name, &sbuf) < )
{
continue;
} char perms[] = "----------";
perms[0] = '?';
//获取文件类型
mode_t mode = sbuf.st_mode;
switch (mode & S_IFMT)
{
case S_IFREG://普通文件
perms[0] = '-';
break;
case S_IFDIR://目录文件
perms[0] = 'd';
break;
case S_IFLNK://链接文件
perms[0] = 'l';
break;
case S_IFIFO://管道文件
perms[0] = 'p';
break;
case S_IFSOCK://套接字文件
perms[0] = 's';
break;
case S_IFCHR://字符设备文件
perms[0] = 'c';
break;
case S_IFBLK://块设备文件
perms[0] = 'b';
break;
}
} return ;
}

接下来要获取的是9位的权限:

也可以通过这些来获取到:

具体代码如下:

int list_common(void)
{
//打开当前目录
DIR *dir = opendir(".");
if (dir == NULL)
{
return ;
} //读取目录并进行遍历
struct dirent *dt;
struct stat sbuf;
while ((dt = readdir(dir)) != NULL)
{ //获取文件的状态
if (lstat(dt->d_name, &sbuf) < )
{
continue;
} char perms[] = "----------";
perms[] = '?';
//获取文件类型
mode_t mode = sbuf.st_mode;
switch (mode & S_IFMT)
{
case S_IFREG://普通文件
perms[] = '-';
break;
case S_IFDIR://目录文件
perms[] = 'd';
break;
case S_IFLNK://链接文件
perms[] = 'l';
break;
case S_IFIFO://管道文件
perms[] = 'p';
break;
case S_IFSOCK://套接字文件
perms[] = 's';
break;
case S_IFCHR://字符设备文件
perms[] = 'c';
break;
case S_IFBLK://块设备文件
perms[] = 'b';
break;
} //获取文件9个权限位
if (mode & S_IRUSR)
{
perms[1] = 'r';
}
if (mode & S_IWUSR)
{
perms[2] = 'w';
}
if (mode & S_IXUSR)
{
perms[3] = 'x';
}
if (mode & S_IRGRP)
{
perms[4] = 'r';
}
if (mode & S_IWGRP)
{
perms[5] = 'w';
}
if (mode & S_IXGRP)
{
perms[6] = 'x';
}
if (mode & S_IROTH)
{
perms[7] = 'r';
}
if (mode & S_IWOTH)
{
perms[8] = 'w';
}
if (mode & S_IXOTH)
{
perms[9] = 'x';
}
} return ;
}

接下来就是一些特珠的权限位,下面来看下:

文件a的权限如上,下面给它更改一下权限:

同样的组权限也有类似的小s,大s,而其它组的权限有小t,大T权限,所以下面来处理这些特珠权限:

int list_common(void)
{
//打开当前目录
DIR *dir = opendir(".");
if (dir == NULL)
{
return ;
} //读取目录并进行遍历
struct dirent *dt;
struct stat sbuf;
while ((dt = readdir(dir)) != NULL)
{ //获取文件的状态
if (lstat(dt->d_name, &sbuf) < )
{
continue;
} char perms[] = "----------";
perms[] = '?';
//获取文件类型
mode_t mode = sbuf.st_mode;
switch (mode & S_IFMT)
{
case S_IFREG://普通文件
perms[] = '-';
break;
case S_IFDIR://目录文件
perms[] = 'd';
break;
case S_IFLNK://链接文件
perms[] = 'l';
break;
case S_IFIFO://管道文件
perms[] = 'p';
break;
case S_IFSOCK://套接字文件
perms[] = 's';
break;
case S_IFCHR://字符设备文件
perms[] = 'c';
break;
case S_IFBLK://块设备文件
perms[] = 'b';
break;
} //获取文件9个权限位
if (mode & S_IRUSR)
{
perms[] = 'r';
}
if (mode & S_IWUSR)
{
perms[] = 'w';
}
if (mode & S_IXUSR)
{
perms[] = 'x';
}
if (mode & S_IRGRP)
{
perms[] = 'r';
}
if (mode & S_IWGRP)
{
perms[] = 'w';
}
if (mode & S_IXGRP)
{
perms[] = 'x';
}
if (mode & S_IROTH)
{
perms[] = 'r';
}
if (mode & S_IWOTH)
{
perms[] = 'w';
}
if (mode & S_IXOTH)
{
perms[] = 'x';
}
//获取特珠权限位
if (mode & S_ISUID)
{
perms[3] = (perms[3] == 'x') ? 's' : 'S';
}
if (mode & S_ISGID)
{
perms[6] = (perms[6] == 'x') ? 's' : 'S';
}
if (mode & S_ISVTX)
{
perms[9] = (perms[9] == 'x') ? 't' : 'T';
}
} return ;
}

并且要将之前获取的权限信息与连接接,以及后面的那些信息连接起来格式化,所以下面将获取的信息格式化:

int list_common(void)
{
//打开当前目录
DIR *dir = opendir(".");
if (dir == NULL)
{
return ;
} //读取目录并进行遍历
struct dirent *dt;
struct stat sbuf;
while ((dt = readdir(dir)) != NULL)
{ //获取文件的状态
if (lstat(dt->d_name, &sbuf) < )
{
continue;
} char perms[] = "----------";
perms[] = '?';
//获取文件类型
mode_t mode = sbuf.st_mode;
switch (mode & S_IFMT)
{
case S_IFREG://普通文件
perms[] = '-';
break;
case S_IFDIR://目录文件
perms[] = 'd';
break;
case S_IFLNK://链接文件
perms[] = 'l';
break;
case S_IFIFO://管道文件
perms[] = 'p';
break;
case S_IFSOCK://套接字文件
perms[] = 's';
break;
case S_IFCHR://字符设备文件
perms[] = 'c';
break;
case S_IFBLK://块设备文件
perms[] = 'b';
break;
} //获取文件9个权限位
if (mode & S_IRUSR)
{
perms[] = 'r';
}
if (mode & S_IWUSR)
{
perms[] = 'w';
}
if (mode & S_IXUSR)
{
perms[] = 'x';
}
if (mode & S_IRGRP)
{
perms[] = 'r';
}
if (mode & S_IWGRP)
{
perms[] = 'w';
}
if (mode & S_IXGRP)
{
perms[] = 'x';
}
if (mode & S_IROTH)
{
perms[] = 'r';
}
if (mode & S_IWOTH)
{
perms[] = 'w';
}
if (mode & S_IXOTH)
{
perms[] = 'x';
}
//获取特珠权限位
if (mode & S_ISUID)
{
perms[] = (perms[] == 'x') ? 's' : 'S';
}
if (mode & S_ISGID)
{
perms[] = (perms[] == 'x') ? 's' : 'S';
}
if (mode & S_ISVTX)
{
perms[] = (perms[] == 'x') ? 't' : 'T';
} //格式化信息
char buf[1024] = {0};
int off = 0;
off += sprintf(buf, "%s ", perms);//连接权限位
off += sprintf(buf + off, " %3d %-8d %-8d ", sbuf.st_nlink, sbuf.st_uid, sbuf.st_gid);//连接连接数、uid、gid
off += sprintf(buf + off, "%8lu ", (unsigned long)sbuf.st_size);//连接文件大小,以8位的长度展现
} return ;
}

这两个日期显示的规则是这样:

下面来具体实现:

首先需要获取系统的当前时间,可以利用这个函数:

【说明】而格式化中的“%b”、“%e”、"%Y"是什么意思,可以查看该函数:strftime:

接下来将日期进行格式化:

最后一个信息则是文件名:

写了这么多,下面来编译运行一下:

一大堆头文件没有包含,所以需要包含一下:

再次编译运行:

编译通过了,下面来编写一下测试代码来验证当前目录输出代码是不OK:

运行看下效果:

再次编译运行:

将编译的可执行文件拷贝到home目录,以便跟vsftptd的进行对比,然后再进入home目录来运行:

修改代码如下:

再次编译运行:

上面来看已经跟vsftpd显示的效果基本一样了,但是有一个链接文件b是指向a的,这里没有显示出来,看下vsftpd是如何显示的:

所以下面得对符号链接文件要进行修改,代码如下:

其中判断一个文件是否是符号链接文件可以用宏S_ISLNK,实际它的内部是:

再次编译运行:

最后跟vsftpd的输出做一下对比:

好了,这次的东西有点多,需好好消化,下次则继续完成FTP目录列表的显示~

27_23

Linux网络编程综合运用之MiniFtp实现(九)的更多相关文章

  1. Linux网络编程综合运用之MiniFtp实现(一)

    春节过后,万物复苏,在这元宵佳节的前一天,决定继续开启新年的学习计划,生命在于运动,提高源于学习,在经过漫长的Linux网络编程学习后,接下来会以一个综合的小项目来将所学的知识点综合运用,首先是对项目 ...

  2. Linux网络编程综合运用之MiniFtp实现(四)

    从今天开始,正式进入MiniFtp的代码编写阶段了,好兴奋,接下来很长一段时间会将整个实现过程从无到有一点点实现出来,达到综合应用的效果,话不多说正入正题: 这节主要是将基础代码框架搭建好,基于上节介 ...

  3. Linux网络编程综合运用之MiniFtp实现(五)

    转眼兴奋的五一小长假就要到来了,在放假前夕还是需要保持一颗淡定的心,上次中已经对miniFTP有基础框架进行了搭建,这次继续进行往上加代码,这次主要还是将经历投射到handle_child()服务进程 ...

  4. Linux网络编程综合运用之MiniFtp实现(八)

    上节中实现了"USER"和"PASS"命令,如下: 事实上FTP是有很多命令组成的,如果就采用上面的这种方法来实现的话,就会有很多if...else if语句, ...

  5. Linux网络编程综合运用之MiniFtp实现(七)

    上节中实现了配置文件的解析,这节来实现用户登录的验证,首先用客户端来登录vsftpd来演示登录的过程: 接着再连接miniftpd,来看下目前的效果: 接下来实现它,与协议相关的模块都是在ftppro ...

  6. Linux网络编程综合运用之MiniFtp实现(六)

    间隔了一周时间没写了,由于今年的股势行情貌似不错的样子,对于对股市完全不懂的我也在蠢蠢欲动,所以最近一周业余时间在“不务正业”-----学习炒股.发现学习它其实挺费神的,满脑子都是走势图,而且是神经有 ...

  7. Linux网络编程综合运用之MiniFtp实现(三)

    前面已经对FTP相关的一些概念有了基本的认识,接下来就要进入代码编写阶段了,也是非常兴奋的阶段,在开启这个它之前先对项目需求进行一个梳理,对其我们要实现的FTP服务器是一个什么样子. ftp命令列表 ...

  8. 【深入浅出Linux网络编程】 "开篇 -- 知其然,知其所以然"

    [深入浅出Linux网络编程]是一个连载博客,内容源于本人的工作经验,旨在给读者提供靠谱高效的学习途径,不必在零散的互联网资源中浪费精力,快速的掌握Linux网络编程. 连载包含4篇,会陆续编写发出, ...

  9. 【linux草鞋应用编程系列】_5_ Linux网络编程

    一.网络通信简介   第一部分内容,暂时没法描述,内容实在太多,待后续专门的系列文章.   二.linux网络通信     在linux中继承了Unix下“一切皆文件”的思想, 在linux中要实现网 ...

随机推荐

  1. Django:bootstrap table自定义查询实现

    参考:https://jalena.bcsytv.com/archives/tag/bootstrap 背景: bootstrap table在客户端分页方式下,自带有简易的搜索功能,但是功能太单一, ...

  2. 高级UI-TableLayout

    TableLayout选项卡,用于需要使用选项卡的场景,一般是用于切换Fragment,现在的主流做法一般是TableLayout+ViewPager+Fragment,综合实现选项卡的操作 由于Ta ...

  3. qt linux下自动检测U盘热插拔

    如果用U盘来更新软件,需要先删除当前的可执行文件,但是如果删除之后,复制失败的话会导致后续没有文件可以运行了.一个方法是将新的可执行文件放到另一个目录中,当copy成功后修改linux的启动文件,然后 ...

  4. Elasticsearch配置安装

    跨域  elasticsearch-head连接es时会提示连接失败,有可能就是没有开启跨域 http.cors.enabled 是否支持跨域,默认为false http.cors.allow-ori ...

  5. 使用TypeScript创建Vue项目

    Vue的灵活性总是让代码看起来非常洗练,对TypeScript来说也是一种挑战, 好在Vue对TypeScript进行了一次全方位的适配. 相对于React严谨的代码,Redux啰嗦的样板代码,Vue ...

  6. 2019ICPC南昌现场赛总结

    非常可惜的一场比赛,多了60分钟罚时与银牌无缘.今年6场ICPC网络赛里面打的最差的就是南昌站,冥冥之中自有天意吧,最后被安排去了南昌. 开场被队友叫去先看的L,说是足球,发现就是简单模拟,就直接上机 ...

  7. python 之 面向对象基础(组合和封装)

    7.4 组合 解决类与类之间代码冗余问题有两种解决方案: 1.继承:描述的是类与类之间,什么是什么的关系 2.组合:描述的是类与类之间的关系,是一种什么有什么的关系 一个类产生的对象,该对象拥有一个属 ...

  8. vue 等比例截图组件,支持缩放和旋转

    <template> <div class="crop-image" :style="wrapStyle"> <img :src= ...

  9. 题解-CTS2019随机立方体

    problem \(\mathtt {loj-3119}\) 题意概要:一个 \(n\times m\times l\) 的立方体,立方体中每个格子上都有一个数,如果某个格子上的数比三维坐标中至少有一 ...

  10. JavaScript中对null和undefined的理解

    前沿: 今天工作中遇到了监视一个变量是undefined,结果判断写的是==null 返回值是true,这个结果引起了我对这两个东西的兴趣. 查询了相关的文章理解并测试了.发现有以下特点: 1.广义上 ...