一、FTP服务器

   FTP,File Transfer Protocol 文件传输协议,FTP协议是一种古老的协议,它出现的比HTTP协议还要早,FTP主要应用于网络空间数据交换操作。ftp在工作时会有两个连接:一是命令连接,这个连接始终存在,只有使用bye命令退出会话时,连接才会断开;二是数据连接,数据传输完成,连接即断开。ftp有两种工作模式,主动模式和被动模式,主动模式工作在tcp/20端口,而被动模式则工作在tcp协议下的随机端口。通常FTP在身份认证时工作在21号端口。

   在红帽系统下,vsftp是一款相对安全的FTP软件,在生产环境下可以使用它快速的实现FTP服务器的搭建。vsftp的用户认证支持两类用户,一是系统用户,二是虚拟用户。系统用户故名思义,就是使用系统上的普通用户作为FTP用户,这种方式操作相对简单。而虚拟用户则是使用一个系统用户映射成所有虚拟用户,访问时的文件目录是为此系统用户家目录。虚拟用户的用户帐号信息的存储分两种方式,一是将用户信息保存于一个hash编码的文件,奇数行为用户名,偶数行为密码;二是,存储于关系型数据库。我们知道,系统上几乎所有的和帐号认证有关的操作都是通过系统的pam模块负责管理,vsftp也不例外,我们使用mysql管理vsftp的帐号密码信息,则可以使用pam的mysql模块。pam默认不支持mysql数据库,因此在使用前需要安装此模块。


二、FTP服务器实现

1、安装软件

# yum -y install vsftpd mysql-server mysql-devel pam_mysql

wKiom1M2PtuxG0yFAAINl1y4T9g787.jpg


wKioL1M2Pr7huOYpAAPWIqZqB0g984.jpg


2、创建数据库

mysql> create database vsftpd;
mysql> grant all on vsftpd.* to 'vsftpuser'@'localhost' identified by 'vsftppass';
mysql> grant all on vsftpd.* to 'vsftpuser'@'127.0.0.1' identified by 'vsftppass';
mysql> grant all on vsftpd.* to 'vsftpuser'@'172.16.36.1' identified by 'vsftppass';
mysql> flush privileges;
mysql> use vsftpd;
mysql> create table users (
    -> id int AUTO_INCREMENT NOT NULL,
    -> name char(20) binary NOT NULL,
    -> password char(48) binary NOT NULL,
    -> primary key(id)
    -> );

wKiom1M2QRqQRIWVAAGjQ5jGDb8745.jpg


2、创建vsftp虚拟用户

insert into users(name,password) values('wu',md5('wu'));
insert into users(name,password) values('json',md5('json'));

wKioL1M2c23QGzNHAAHLRJn3msU047.jpg


3、建立pam认证所需文件

#vi /etc/pam.d/vsftpd.mysql

添加如下两行:

auth required /lib64/security/pam_mysql.so user=vsftpuser passwd=vsftppass host=172.16.36.1 db=vsftpd table=users usercolumn=name passwdcolumn=password crypt=3
account required /lib64/security/pam_mysql.so user=vsftpuser passwd=vsftppass host=172.16.36.1 db=vsftpd table=users usercolumn=name passwdcolumn=password crypt=3

user: 数据库用户名

passwd: 数据库用户密码

host: 为数据库主机IP地址,如果是编译安装的数据库,只能填写远程IP地址,并给把这个IP地址授权给用户

db: 数据库名

table: 表名

usercolumn: 用户字段名

passwdcolumn: 用户密码字段名

crypt: 密码保存方式,0为明文,1为encrypt函数加密,2为password函数加密,3为MD5加密


注意:由于mysql的安装方式不同,pam_mysql.so基于unix sock连接mysql服务器时可能会出问题,此时,建议授权一个可远程连接的mysql并访问vsftpd数据库的用户。


4、建立虚拟用户映射的系统用户及对应的目录

# useradd -s /sbin/nologin -d /var/ftproot vuser
# chmod go+rx /var/ftproot


5、请确保/etc/vsftpd/vsftpd.conf中已经启用了以下选项

anonymous_enable=NO   #禁止匿名用户登录
local_enable=YES      #允许本地用户登录
write_enable=YES      #本地用户有写权限
anon_upload_enable=NO #匿名用户没有上传权限
anon_mkdir_write_enable=NO #匿名用户没有创建权限
chroot_local_user=YES #禁锢用户家目录

而后添加以下选项

guest_enable=YES
guest_username=vuser

并确保pam_service_name选项的值如下所示

pam_service_name=vsftpd.mysql


6、重新启动服务,测试一下。

service vsftpd restart

打开CMD程序远程测试一下:

wKioL1M2d5Lx7S7HAAIuIbAIAh0966.jpg


两个帐号都能成功登录!


二、通过web管理VSFTP

   使用mysql管理vsftp用户的一个好处是,当服务器数据迁移时,只要把相关的数据库打包备份导入新的数据库就行了,管理相当方便。如果本机上有web服务器的话,我们还可以使用PHP开发一个web界面专门用来管理vsftp。

例如:我们新建一个虚拟主机:

<VirtualHost *:80>
    DocumentRoot "/var/www/html/vsftpadmin"
    ServerName ftpadmin.wubinary.com
    ServerAlias ftpadmin.wubinary.com
    ErrorLog "logs/ftpadmin.wubinary.com.error_log"
    CustomLog "logs/ftpadmin.wubinary.com.access_log" common
    ProxyRequests Off
    ProxyPassMatch ^/(.*\.php)$ fcgi://127.0.0.1:9000/var/www/html/vsftpadmin/$1
    <Directory "/var/www/html/vsftpadmin">
      Options None
      AllowOverride AuthConfig
      AuthType Basic
      AuthName "Vsftp Admin"
      AuthUserFile /etc/httpd/extra/.htpasswd
      Require valid-user
    </Directory>
</VirtualHost>

对这个域名作身份认证,或者使用其它的方式对用户进行限制,用PHP写一个管理页面放在相关的目录下,我们就可以在web界面下管理我们的FTP服务器帐号了。

wKioL1M2ilfAVTAOAAHhbRmXfQg613.jpg


wKiom1M2io-wsfBAAAFA9QYAVe8057.jpg


整个FTP服务器的搭建工作到此就完成了。vsftp web管理界面的PHP源码如下:


vsftpadmin.php


<?php
header("Content-Type:text/html;charset=utf-8");
error_reporting(E_ALL & ~ E_NOTICE);
//提示窗
function alertExit($msg,$flush=0){
    // echo "<script language='javascript'>alert($msg);</script>";
    echo "<script type='text/javascript' language='javascript'>alert('$msg');</script>";
    if ($flush == 1) {
        echo "<script type='text/javascript' language='javascript'>window.location.href='vsftpadmin.php'</script>";
    }elseif ($flush == 2) {
        echo "<script type='text/javascript' language='javascript'>history.back();self.location.reload();</script>";
    }
}
//输出头部
function htmlheader($title){
echo <<<EOF
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type"  content="text/html; charset=utf-8">
<title>{$title}</title>
<meta name="description" content="vsftp管理工具">
<meta name="keywords" content="linux,ftp,vsftp,mysql,pam_mysql,php">
<style type="text/css">
*{margin:0; padding:0; font-size:12px;}
ul,li{ list-style:none;}
a {text-decoration:none;color:#37a}
a:hover { background:#37a;color:white;padding:2px;}
img { border:0;}
.clear{ clear:both;}
.right{ float:right;}
.left{ float:left;}
#content{width:560px;margin:0 auto;text-align:center;margin-top:40px;}
.admin{border:1px solid #ccc; padding:30px;text-align:left;}
.admin h4{margin:10px 0;}
.admin p{margin:10px 0;}
.admin p .btn{padding:2px 4px;}
</style>
</head>
<body>
<div id="content">
    <div class="admin">
        <h4>{$title}</h4>
        <p class="admin_btn">管理菜单:<a href="?ac=ftp">ftp状态</a>&nbsp;&nbsp;<a href="vsftpadmin.php">用户列表</a>&nbsp;&nbsp;<a href="?ac=add">添加用户</a></p>
EOF;
}
//输出尾部
function htmlfooter(){
echo <<<EOF
    </div>
</div>
</body>
</html>
EOF;
}
//查询
function query($sql){
                
    $result = mysql_query($sql)  or die ("SQL语句查询错误: " . mysql_error());
    if (mysql_num_rows($result) == 0) {
            die('SQL: '.$sql.'<br>未查询到相关数据');
    }
    $arrReturn = array();
    $index = 0;
    while($arr = mysql_fetch_assoc($result)){
        $arrReturn[$index] = $arr;
        $index++;
    }
    return $arrReturn;
}
//添加
function adduser($name,$password){
                
    if (strlen($name)<3) {
        alertExit('用户名至少三位!',1);
    }
    if (strlen($password)<6) {
        alertExit('密码必须大于八位!',1);
    }
    $sql = "select * from users where name='$name'";
    $result = mysql_query($sql)  or die ("SQL语句查询错误: " . mysql_error());
    if (mysql_num_rows($result) > 0) {
            alertExit('用户已存在',1);
    }else{
        $sql = "insert into users(name,password) values('$name',md5('$password'))";
        $result = mysql_query($sql)  or die ("添加角户出错: " . mysql_error());
        // echo mysql_affected_rows();
        if(mysql_affected_rows()==1){
            alertExit("添加用户成功!",2);
        }
    }  
}
//删除
function deluser($id){
    $sql = 'select * from users where id='.$id;
    $result = mysql_query($sql)  or die ("SQL语句查询错误: " . mysql_error());
    if (mysql_num_rows($result) == 0) {
            alertExit('用户不存在',1);
    }else{
        $sql = 'delete from users where id='.$id;
        $result = mysql_query($sql)  or die ("删除角户出错: " . mysql_error());
        if(mysql_affected_rows()==1){
            alertExit("删除用户成功!",1);
        }
    }
}
//修改
function moduser($id,$name,$password){
    $sql = "select * from users where id=$id";
    $result = mysql_query($sql)  or die ("SQL语句查询错误: " . mysql_error());
    if (mysql_num_rows($result) == 0) {
            alertExit('用户不存在',1);
    }else{
        $sql = "select * from users where name='$name' and id!=$id";
        $result = mysql_query($sql)  or die ("SQL语句查询错误: " . mysql_error());
        if (mysql_num_rows($result) > 0) {
                alertExit('用户已存在',1);
        }else{     
            $sql = "update users set name='$name', password='$password' where id=$id";
            $result = mysql_query($sql)  or die ("修改角户出错: " . mysql_error());
            if(mysql_affected_rows()==1){
                alertExit("修改用户成功!",1);
            }
            alertExit("未作任何操作!",1);
        }
    }
}
//ftp状态管理
function ftpadmin($service='status'){
                
    $arrFtp = array();
    $result = mysql_query("SELECT id FROM users");
    $num_rows = mysql_num_rows($result);
    $arrFtp['usercount'] = $num_rows;
                
    if($service=='status'){
                    
        $arrFtp['status'] = `service vsftpd status`;
                    
    }elseif($service=='restart'){
                    
        $arrFtp['status'] = `service vsftpd restart`;
                    
    }elseif ($service=='stop') {
                    
        $arrFtp['status'] = `service vsftpd stop`;
                    
    }
                
    if(empty($arrFtp['status'])){
        $arrFtp['status'] = 'Unknow';
    }
    return $arrFtp;
                
}
$conn = @mysql_connect('127.0.0.1','root','')  or die ('数据库连接错误: ' . mysql_error());
mysql_select_db('vsftp', $conn) or die ('选择数据库错误: ' . mysql_error());
$strAction = htmlspecialchars($_GET['ac']);
$arrAction = array('del', 'mod', 'add', 'ftp');
if (! in_array($strAction, $arrAction)) {
        htmlheader('Vsftp 管理');
        $arrUserList = query('select * from users');
        foreach ($arrUserList as $key => $value) {
        ?>
        <form action='./vsftpadmin.php?ac=mod' method='post'>
        <p><label>name:</label> <input type="text" name="name" value="<?php echo $value['name'];?>">
            <label>password:</label> <input type="password" name="password" value="<?php echo $value["password"];?>">
            <input type='hidden' name='id' value="<?php echo $value["id"];?>">
             <input type="submit" value="修改" class="btn">&nbsp;<input type="button" οnclick="window.location.href='?id=<?php echo $value["id"];?>&ac=del'" value="删除" class="btn"></p>
        </form>
        <?php }
        htmlfooter();
}else{
                
    if ($strAction=='add') {
                    
        if ($_SERVER['REQUEST_METHOD']=='POST') {
                        
            $name = htmlspecialchars($_POST['name']);
            $password = htmlspecialchars($_POST['password']);
                        
            adduser($name,$password);
                        
        }else{
            htmlheader('添加用户');
            ?>
        <form action='./vsftpadmin.php?ac=add' method='post'>
        <p><label>name:</label> <input type="text" name="name" value="">
            <label>password:</label> <input type="password" name="password" value="">
            <input type="submit" value="提交" class="btn">&nbsp;<input type="reset" value="重置" class="btn"></p>
        </form>
<?php       
            htmlfooter();
        }
    }elseif ($strAction=='del') {
                    
        $intId = intval($_GET['id']);
                    
        if($intId!=''){
            deluser($intId);
        }else{
            alertExit('参数错误!',1);
        }
                    
    }elseif ($strAction=='mod') {
                    
        $intId = intval($_POST['id']);
                    
        $name = htmlspecialchars($_POST['name']);
        $password = htmlspecialchars($_POST['password']);
                                
        if($name!='' && $password!=''){
            moduser($intId,$name,$password);
        }else{
            alertExit('参数错误!',1);
        }
    }elseif($strAction=='ftp'){
            htmlheader('FTP状态');
            $arrFtp = array();
            $status = htmlspecialchars($_GET['status']);
            if($status=='restart')$arrFtp = ftpadmin('restart');
            elseif($status=='stop')$arrFtp = ftpadmin('stop');
            else $arrFtp = ftpadmin();
            ?>
        <p><label>用户总数:</label> <?php echo $arrFtp['usercount'];?> <br>
            <label>vsftp状态:</label> <?php echo $arrFtp['status'];?>
            <input type="button" οnclick="window.location.href='?ac=ftp&status=restart'" value="重启" class="btn">&nbsp;<input type="button" οnclick="window.location.href='?ac&status=stop'" value="停止" class="btn"></p>
<?php       
            htmlfooter();
                    
    }else{
        die('无效的地址!');
    }
}