国产无遮挡裸体免费直播视频,久久精品国产蜜臀av,动漫在线视频一区二区,欧亚日韩一区二区三区,久艹在线 免费视频,国产精品美女网站免费,正在播放 97超级视频在线观看,斗破苍穹年番在线观看免费,51最新乱码中文字幕

使用PHP如何實現(xiàn)高效安全的ftp服務器(二)

 更新時間:2015年12月30日 09:31:18   作者:Terry.Zhang  
這篇文章主要介紹了使用PHP如何實現(xiàn)高效安全的ftp服務器(二)的相關資料,需要的朋友可以參考下

在上篇文章給大家介紹了使用PHP如何實現(xiàn)高效安全的ftp服務器(一),感興趣的朋友可以點擊了解詳情。接下來通過本篇文章給大家介紹使用PHP如何實現(xiàn)高效安全的ftp服務器(二),具體內(nèi)容如下所示:

1.實現(xiàn)用戶類CUser。

  用戶的存儲采用文本形式,將用戶數(shù)組進行json編碼?! ?br />

用戶文件格式:

* array(
* 'user1' => array(
* 'pass'=>'',
* 'group'=>'',
* 'home'=>'/home/ftp/', //ftp主目錄
* 'active'=>true,
* 'expired=>'2015-12-12',
* 'description'=>'',
* 'email' => '',
* 'folder'=>array(
* //可以列出主目錄下的文件和目錄,但不能創(chuàng)建和刪除,也不能進入主目錄下的目錄
* //前1-5位是文件權限,6-9是文件夾權限,10是否繼承(inherit)
* array('path'=>'/home/ftp/','access'=>'RWANDLCNDI'),
* //可以列出/home/ftp/a/下的文件和目錄,可以創(chuàng)建和刪除,可以進入/home/ftp/a/下的子目錄,可以創(chuàng)建和刪除。
* array('path'=>'/home/ftp/a/','access'=>'RWAND-----'),
* ),
* 'ip'=>array(
* 'allow'=>array(ip1,ip2,...),//支持*通配符: 192.168.0.*
* 'deny'=>array(ip1,ip2,...)
* )
* ) 
* )
* 
* 組文件格式:
* array(
* 'group1'=>array(
* 'home'=>'/home/ftp/dept1/',
* 'folder'=>array(
* 
* ),
* 'ip'=>array(
* 'allow'=>array(ip1,ip2,...),
* 'deny'=>array(ip1,ip2,...)
* )
* )
* ) 

  文件夾和文件的權限說明:

* 文件權限
* R讀 : 允許用戶讀取(即下載)文件。該權限不允許用戶列出目錄內(nèi)容,執(zhí)行該操作需要列表權限。
* W寫: 允許用戶寫入(即上傳)文件。該權限不允許用戶修改現(xiàn)有的文件,執(zhí)行該操作需要追加權限。
* A追加: 允許用戶向現(xiàn)有文件中追加數(shù)據(jù)。該權限通常用于使用戶能夠對部分上傳的文件進行續(xù)傳。
* N重命名: 允許用戶重命名現(xiàn)有的文件。
* D刪除: 允許用戶刪除文件。
*
* 目錄權限
* L列表: 允許用戶列出目錄中包含的文件。
* C創(chuàng)建: 允許用戶在目錄中新建子目錄。
* N重命名: 允許用戶在目錄中重命名現(xiàn)有子目錄。
* D刪除: 允許用戶在目錄中刪除現(xiàn)有子目錄。注意: 如果目錄包含文件,用戶要刪除目錄還需要具有刪除文件權限。
*
* 子目錄權限
* I繼承: 允許所有子目錄繼承其父目錄具有的相同權限。繼承權限適用于大多數(shù)情況,但是如果訪問必須受限于子文件夾,例如實施強制訪問控制(Mandatory Access Control)時,則取消繼承并為文件夾逐一授予權限。
*

  實現(xiàn)代碼如下:  

class User{
const I = 1; // inherit
const FD = 2; // folder delete
const FN = 4; // folder rename
const FC = 8; // folder create
const FL = 16; // folder list
const D = 32; // file delete
const N = 64; // file rename
const A = 128; // file append
const W = 256; // file write (upload)
const R = 512; // file read (download) 
private $hash_salt = '';
private $user_file;
private $group_file;
private $users = array();
private $groups = array();
private $file_hash = ''; 
public function __construct(){
$this->user_file = BASE_PATH.'/conf/users';
$this->group_file = BASE_PATH.'/conf/groups';
$this->reload();
}
/**
* 返回權限表達式
* @param int $access
* @return string
*/
public static function AC($access){
$str = '';
$char = array('R','W','A','N','D','L','C','N','D','I');
for($i = 0; $i < 10; $i++){
if($access & pow(2,9-$i))$str.= $char[$i];else $str.= '-';
}
return $str;
}
/**
* 加載用戶數(shù)據(jù)
*/
public function reload(){
$user_file_hash = md5_file($this->user_file);
$group_file_hash = md5_file($this->group_file); 
if($this->file_hash != md5($user_file_hash.$group_file_hash)){
if(($user = file_get_contents($this->user_file)) !== false){
$this->users = json_decode($user,true);
if($this->users){
//folder排序
foreach ($this->users as $user=>$profile){
if(isset($profile['folder'])){
$this->users[$user]['folder'] = $this->sortFolder($profile['folder']);
}
}
}
}
if(($group = file_get_contents($this->group_file)) !== false){
$this->groups = json_decode($group,true);
if($this->groups){
//folder排序
foreach ($this->groups as $group=>$profile){ 
if(isset($profile['folder'])){ 
$this->groups[$group]['folder'] = $this->sortFolder($profile['folder']);
}
}
}
}
$this->file_hash = md5($user_file_hash.$group_file_hash); 
}
}
/**
* 對folder進行排序
* @return array
*/
private function sortFolder($folder){
uasort($folder, function($a,$b){
return strnatcmp($a['path'], $b['path']);
}); 
$result = array();
foreach ($folder as $v){
$result[] = $v;
} 
return $result;
}
/**
* 保存用戶數(shù)據(jù)
*/
public function save(){
file_put_contents($this->user_file, json_encode($this->users),LOCK_EX);
file_put_contents($this->group_file, json_encode($this->groups),LOCK_EX);
}
/**
* 添加用戶
* @param string $user
* @param string $pass
* @param string $home
* @param string $expired
* @param boolean $active
* @param string $group
* @param string $description
* @param string $email
* @return boolean
*/
public function addUser($user,$pass,$home,$expired,$active=true,$group='',$description='',$email = ''){
$user = strtolower($user);
if(isset($this->users[$user]) || empty($user)){
return false;
} 
$this->users[$user] = array(
'pass' => md5($user.$this->hash_salt.$pass),
'home' => $home,
'expired' => $expired,
'active' => $active,
'group' => $group,
'description' => $description,
'email' => $email,
);
return true;
}
/**
* 設置用戶資料
* @param string $user
* @param array $profile
* @return boolean
*/
public function setUserProfile($user,$profile){
$user = strtolower($user);
if(is_array($profile) && isset($this->users[$user])){
if(isset($profile['pass'])){
$profile['pass'] = md5($user.$this->hash_salt.$profile['pass']);
}
if(isset($profile['active'])){
if(!is_bool($profile['active'])){
$profile['active'] = $profile['active'] == 'true' ? true : false;
}
} 
$this->users[$user] = array_merge($this->users[$user],$profile);
return true;
}
return false;
}
/**
* 獲取用戶資料
* @param string $user
* @return multitype:|boolean
*/
public function getUserProfile($user){
$user = strtolower($user);
if(isset($this->users[$user])){
return $this->users[$user];
}
return false;
}
/**
* 刪除用戶
* @param string $user
* @return boolean
*/
public function delUser($user){
$user = strtolower($user);
if(isset($this->users[$user])){
unset($this->users[$user]);
return true;
}
return false;
}
/**
* 獲取用戶列表
* @return array
*/
public function getUserList(){
$list = array();
if($this->users){
foreach ($this->users as $user=>$profile){
$list[] = $user;
}
}
sort($list);
return $list;
}
/**
* 添加組
* @param string $group
* @param string $home
* @return boolean
*/
public function addGroup($group,$home){
$group = strtolower($group);
if(isset($this->groups[$group])){
return false;
}
$this->groups[$group] = array(
'home' => $home
);
return true;
}
/**
* 設置組資料
* @param string $group
* @param array $profile
* @return boolean
*/
public function setGroupProfile($group,$profile){
$group = strtolower($group);
if(is_array($profile) && isset($this->groups[$group])){
$this->groups[$group] = array_merge($this->groups[$group],$profile);
return true;
}
return false;
}
/**
* 獲取組資料
* @param string $group
* @return multitype:|boolean
*/
public function getGroupProfile($group){
$group = strtolower($group);
if(isset($this->groups[$group])){
return $this->groups[$group];
}
return false;
}
/**
* 刪除組
* @param string $group
* @return boolean
*/
public function delGroup($group){
$group = strtolower($group);
if(isset($this->groups[$group])){
unset($this->groups[$group]);
foreach ($this->users as $user => $profile){
if($profile['group'] == $group)
$this->users[$user]['group'] = '';
}
return true;
}
return false;
}
/**
* 獲取組列表
* @return array
*/
public function getGroupList(){
$list = array();
if($this->groups){
foreach ($this->groups as $group=>$profile){
$list[] = $group;
}
}
sort($list);
return $list;
}
/**
* 獲取組用戶列表
* @param string $group
* @return array
*/
public function getUserListOfGroup($group){
$list = array();
if(isset($this->groups[$group]) && $this->users){
foreach ($this->users as $user=>$profile){
if(isset($profile['group']) && $profile['group'] == $group){
$list[] = $user;
}
}
}
sort($list);
return $list;
}
/**
* 用戶驗證
* @param string $user
* @param string $pass
* @param string $ip
* @return boolean
*/
public function checkUser($user,$pass,$ip = ''){
$this->reload();
$user = strtolower($user);
if(isset($this->users[$user])){
if($this->users[$user]['active'] && time() <= strtotime($this->users[$user]['expired'])
&& $this->users[$user]['pass'] == md5($user.$this->hash_salt.$pass)){
if(empty($ip)){
return true;
}else{
//ip驗證
return $this->checkIP($user, $ip);
}
}else{
return false;
} 
}
return false;
}
/**
* basic auth 
* @param string $base64 
*/
public function checkUserBasicAuth($base64){
$base64 = trim(str_replace('Basic ', '', $base64));
$str = base64_decode($base64);
if($str !== false){
list($user,$pass) = explode(':', $str,2);
$this->reload();
$user = strtolower($user);
if(isset($this->users[$user])){
$group = $this->users[$user]['group'];
if($group == 'admin' && $this->users[$user]['active'] && time() <= strtotime($this->users[$user]['expired'])
&& $this->users[$user]['pass'] == md5($user.$this->hash_salt.$pass)){ 
return true;
}else{
return false;
}
}
}
return false;
}
/**
* 用戶登錄ip驗證
* @param string $user
* @param string $ip
* 
* 用戶的ip權限繼承組的IP權限。
* 匹配規(guī)則:
* 1.進行組允許列表匹配;
* 2.如同通過,進行組拒絕列表匹配;
* 3.進行用戶允許匹配
* 4.如果通過,進行用戶拒絕匹配
* 
*/
public function checkIP($user,$ip){
$pass = false;
//先進行組驗證 
$group = $this->users[$user]['group'];
//組允許匹配
if(isset($this->groups[$group]['ip']['allow'])){
foreach ($this->groups[$group]['ip']['allow'] as $addr){
$pattern = '/'.str_replace('*','\d+',str_replace('.', '\.', $addr)).'/';
if(preg_match($pattern, $ip) && !empty($addr)){
$pass = true;
break;
}
}
}
//如果允許通過,進行拒絕匹配
if($pass){
if(isset($this->groups[$group]['ip']['deny'])){
foreach ($this->groups[$group]['ip']['deny'] as $addr){
$pattern = '/'.str_replace('*','\d+',str_replace('.', '\.', $addr)).'/';
if(preg_match($pattern, $ip) && !empty($addr)){
$pass = false;
break;
}
}
}
}
if(isset($this->users[$user]['ip']['allow'])){ 
foreach ($this->users[$user]['ip']['allow'] as $addr){
$pattern = '/'.str_replace('*','\d+',str_replace('.', '\.', $addr)).'/';
if(preg_match($pattern, $ip) && !empty($addr)){
$pass = true;
break;
}
}
}
if($pass){
if(isset($this->users[$user]['ip']['deny'])){
foreach ($this->users[$user]['ip']['deny'] as $addr){
$pattern = '/'.str_replace('*','\d+',str_replace('.', '\.', $addr)).'/';
if(preg_match($pattern, $ip) && !empty($addr)){
$pass = false;
break;
}
}
}
}
echo date('Y-m-d H:i:s')." [debug]\tIP ACCESS:".' '.($pass?'true':'false')."\n";
return $pass;
}
/**
* 獲取用戶主目錄
* @param string $user
* @return string
*/
public function getHomeDir($user){
$user = strtolower($user);
$group = $this->users[$user]['group'];
$dir = '';
if($group){
if(isset($this->groups[$group]['home']))$dir = $this->groups[$group]['home'];
}
$dir = !empty($this->users[$user]['home'])?$this->users[$user]['home']:$dir;
return $dir;
}
//文件權限判斷
public function isReadable($user,$path){ 
$result = $this->getPathAccess($user, $path);
if($result['isExactMatch']){
return $result['access'][0] == 'R';
}else{
return $result['access'][0] == 'R' && $result['access'][9] == 'I';
}
} 
public function isWritable($user,$path){ 
$result = $this->getPathAccess($user, $path); 
if($result['isExactMatch']){
return $result['access'][1] == 'W';
}else{
return $result['access'][1] == 'W' && $result['access'][9] == 'I';
}
}
public function isAppendable($user,$path){
$result = $this->getPathAccess($user, $path);
if($result['isExactMatch']){
return $result['access'][2] == 'A';
}else{
return $result['access'][2] == 'A' && $result['access'][9] == 'I';
}
} 
public function isRenamable($user,$path){
$result = $this->getPathAccess($user, $path);
if($result['isExactMatch']){
return $result['access'][3] == 'N';
}else{
return $result['access'][3] == 'N' && $result['access'][9] == 'I';
}
}
public function isDeletable($user,$path){ 
$result = $this->getPathAccess($user, $path);
if($result['isExactMatch']){
return $result['access'][4] == 'D';
}else{
return $result['access'][4] == 'D' && $result['access'][9] == 'I';
}
}
//目錄權限判斷
public function isFolderListable($user,$path){
$result = $this->getPathAccess($user, $path);
if($result['isExactMatch']){
return $result['access'][5] == 'L';
}else{
return $result['access'][5] == 'L' && $result['access'][9] == 'I';
}
}
public function isFolderCreatable($user,$path){
$result = $this->getPathAccess($user, $path);
if($result['isExactMatch']){
return $result['access'][6] == 'C';
}else{
return $result['access'][6] == 'C' && $result['access'][9] == 'I';
}
}
public function isFolderRenamable($user,$path){
$result = $this->getPathAccess($user, $path);
if($result['isExactMatch']){
return $result['access'][7] == 'N';
}else{
return $result['access'][7] == 'N' && $result['access'][9] == 'I';
}
}
public function isFolderDeletable($user,$path){
$result = $this->getPathAccess($user, $path);
if($result['isExactMatch']){
return $result['access'][8] == 'D';
}else{
return $result['access'][8] == 'D' && $result['access'][9] == 'I';
}
}
/**
* 獲取目錄權限
* @param string $user
* @param string $path
* @return array
* 進行最長路徑匹配
* 
* 返回:
* array(
* 'access'=>目前權限 
* ,'isExactMatch'=>是否精確匹配
* 
* );
* 
* 如果精確匹配,則忽略inherit.
* 否則應判斷是否繼承父目錄的權限,
* 權限位表:
* +---+---+---+---+---+---+---+---+---+---+
* | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 |
* +---+---+---+---+---+---+---+---+---+---+
* | R | W | A | N | D | L | C | N | D | I |
* +---+---+---+---+---+---+---+---+---+---+
* | FILE | FOLDER |
* +-------------------+-------------------+
*/
public function getPathAccess($user,$path){
$this->reload();
$user = strtolower($user);
$group = $this->users[$user]['group']; 
//去除文件名稱
$path = str_replace(substr(strrchr($path, '/'),1),'',$path);
$access = self::AC(0); 
$isExactMatch = false;
if($group){
if(isset($this->groups[$group]['folder'])){ 
foreach ($this->groups[$group]['folder'] as $f){
//中文處理
$t_path = iconv('UTF-8','GB18030',$f['path']); 
if(strpos($path, $t_path) === 0){
$access = $f['access']; 
$isExactMatch = ($path == $t_path?true:false);
} 
}
}
}
if(isset($this->users[$user]['folder'])){
foreach ($this->users[$user]['folder'] as $f){
//中文處理
$t_path = iconv('UTF-8','GB18030',$f['path']);
if(strpos($path, $t_path) === 0){
$access = $f['access']; 
$isExactMatch = ($path == $t_path?true:false);
}
}
}
echo date('Y-m-d H:i:s')." [debug]\tACCESS:$access ".' '.($isExactMatch?'1':'0')." $path\n";
return array('access'=>$access,'isExactMatch'=>$isExactMatch);
} 
/**
* 添加在線用戶
* @param ShareMemory $shm
* @param swoole_server $serv
* @param unknown $user
* @param unknown $fd
* @param unknown $ip
* @return Ambigous <multitype:, boolean, mixed, multitype:unknown number multitype:Ambigous <unknown, number> >
*/
public function addOnline(ShareMemory $shm ,$serv,$user,$fd,$ip){
$shm_data = $shm->read();
if($shm_data !== false){
$shm_data['online'][$user.'-'.$fd] = array('ip'=>$ip,'time'=>time());
$shm_data['last_login'][] = array('user' => $user,'ip'=>$ip,'time'=>time());
//清除舊數(shù)據(jù)
if(count($shm_data['last_login'])>30)array_shift($shm_data['last_login']);
$list = array();
foreach ($shm_data['online'] as $k =>$v){
$arr = explode('-', $k);
if($serv->connection_info($arr[1]) !== false){
$list[$k] = $v;
}
}
$shm_data['online'] = $list;
$shm->write($shm_data);
}
return $shm_data;
}
/**
* 添加登陸失敗記錄
* @param ShareMemory $shm
* @param unknown $user
* @param unknown $ip
* @return Ambigous <number, multitype:, boolean, mixed>
*/
public function addAttempt(ShareMemory $shm ,$user,$ip){
$shm_data = $shm->read();
if($shm_data !== false){
if(isset($shm_data['login_attempt'][$ip.'||'.$user]['count'])){
$shm_data['login_attempt'][$ip.'||'.$user]['count'] += 1;
}else{
$shm_data['login_attempt'][$ip.'||'.$user]['count'] = 1;
}
$shm_data['login_attempt'][$ip.'||'.$user]['time'] = time();
//清除舊數(shù)據(jù)
if(count($shm_data['login_attempt'])>30)array_shift($shm_data['login_attempt']);
$shm->write($shm_data);
}
return $shm_data;
}
/**
* 密碼錯誤上限
* @param unknown $shm
* @param unknown $user
* @param unknown $ip
* @return boolean
*/
public function isAttemptLimit(ShareMemory $shm,$user,$ip){
$shm_data = $shm->read();
if($shm_data !== false){
if(isset($shm_data['login_attempt'][$ip.'||'.$user]['count'])){
if($shm_data['login_attempt'][$ip.'||'.$user]['count'] > 10 &&
time() - $shm_data['login_attempt'][$ip.'||'.$user]['time'] < 600){ 
return true;
}
}
}
return false;
}
/**
* 生成隨機密鑰
* @param int $len
* @return Ambigous <NULL, string>
*/
public static function genPassword($len){
$str = null;
$strPol = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyz@!#$%*+-";
$max = strlen($strPol)-1;
for($i=0;$i<$len;$i++){
$str.=$strPol[rand(0,$max)];//rand($min,$max)生成介于min和max兩個數(shù)之間的一個隨機整數(shù)
}
return $str;
} 
} 

2.共享內(nèi)存操作類

  這個相對簡單,使用php的shmop擴展即可。

class ShareMemory{
private $mode = 0644;
private $shm_key;
private $shm_size;
/**
* 構造函數(shù) 
*/
public function __construct(){
$key = 'F';
$size = 1024*1024;
$this->shm_key = ftok(__FILE__,$key);
$this->shm_size = $size + 1;
}
/**
* 讀取內(nèi)存數(shù)組
* @return array|boolean
*/
public function read(){
if(($shm_id = shmop_open($this->shm_key,'c',$this->mode,$this->shm_size)) !== false){
$str = shmop_read($shm_id,1,$this->shm_size-1);
shmop_close($shm_id);
if(($i = strpos($str,"\0")) !== false)$str = substr($str,0,$i);
if($str){
return json_decode($str,true);
}else{
return array();
}
}
return false;
}
/**
* 寫入數(shù)組到內(nèi)存
* @param array $arr
* @return int|boolean
*/
public function write($arr){
if(!is_array($arr))return false;
$str = json_encode($arr)."\0";
if(strlen($str) > $this->shm_size) return false;
if(($shm_id = shmop_open($this->shm_key,'c',$this->mode,$this->shm_size)) !== false){ 
$count = shmop_write($shm_id,$str,1);
shmop_close($shm_id);
return $count;
}
return false;
}
/**
* 刪除內(nèi)存塊,下次使用時將重新開辟內(nèi)存塊
* @return boolean
*/
public function delete(){
if(($shm_id = shmop_open($this->shm_key,'c',$this->mode,$this->shm_size)) !== false){
$result = shmop_delete($shm_id);
shmop_close($shm_id);
return $result;
}
return false;
}
} 

3.內(nèi)置的web服務器類

  這個主要是嵌入在ftp的http服務器類,功能不是很完善,進行ftp的管理還是可行的。不過需要注意的是,這個實現(xiàn)與apache等其他http服務器運行的方式可能有所不同。代碼是駐留內(nèi)存的。

class CWebServer{
protected $buffer_header = array();
protected $buffer_maxlen = 65535; //最大POST尺寸
const DATE_FORMAT_HTTP = 'D, d-M-Y H:i:s T';
const HTTP_EOF = "\r\n\r\n";
const HTTP_HEAD_MAXLEN = 8192; //http頭最大長度不得超過2k
const HTTP_POST_MAXLEN = 1048576;//1m
const ST_FINISH = 1; //完成,進入處理流程
const ST_WAIT = 2; //等待數(shù)據(jù)
const ST_ERROR = 3; //錯誤,丟棄此包
private $requsts = array();
private $config = array();
public function log($msg,$level = 'debug'){
echo date('Y-m-d H:i:s').' ['.$level."]\t" .$msg."\n";
}
public function __construct($config = array()){
$this->config = array(
'wwwroot' => __DIR__.'/wwwroot/',
'index' => 'index.php',
'path_deny' => array('/protected/'), 
); 
}
public function onReceive($serv,$fd,$data){ 
$ret = $this->checkData($fd, $data);
switch ($ret){
case self::ST_ERROR:
$serv->close($fd);
$this->cleanBuffer($fd);
$this->log('Recevie error.');
break;
case self::ST_WAIT: 
$this->log('Recevie wait.');
return;
default:
break;
}
//開始完整的請求
$request = $this->requsts[$fd];
$info = $serv->connection_info($fd); 
$request = $this->parseRequest($request);
$request['remote_ip'] = $info['remote_ip'];
$response = $this->onRequest($request);
$output = $this->parseResponse($request,$response);
$serv->send($fd,$output);
if(isset($request['head']['Connection']) && strtolower($request['head']['Connection']) == 'close'){
$serv->close($fd);
}
unset($this->requsts[$fd]);
$_REQUEST = $_SESSION = $_COOKIE = $_FILES = $_POST = $_SERVER = $_GET = array();
}
/**
* 處理請求
* @param array $request
* @return array $response
* 
* $request=array(
* 'time'=>
* 'head'=>array(
* 'method'=>
* 'path'=>
* 'protocol'=>
* 'uri'=>
* //other http header
* '..'=>value
* )
* 'body'=>
* 'get'=>(if appropriate)
* 'post'=>(if appropriate)
* 'cookie'=>(if appropriate)
* 
* 
* )
*/
public function onRequest($request){ 
if($request['head']['path'][strlen($request['head']['path']) - 1] == '/'){
$request['head']['path'] .= $this->config['index'];
}
$response = $this->process($request);
return $response;
} 
/**
* 清除數(shù)據(jù)
* @param unknown $fd
*/
public function cleanBuffer($fd){
unset($this->requsts[$fd]);
unset($this->buffer_header[$fd]);
}
/**
* 檢查數(shù)據(jù)
* @param unknown $fd
* @param unknown $data
* @return string
*/
public function checkData($fd,$data){
if(isset($this->buffer_header[$fd])){
$data = $this->buffer_header[$fd].$data;
}
$request = $this->checkHeader($fd, $data);
//請求頭錯誤
if($request === false){
$this->buffer_header[$fd] = $data;
if(strlen($data) > self::HTTP_HEAD_MAXLEN){
return self::ST_ERROR;
}else{
return self::ST_WAIT;
}
}
//post請求檢查
if($request['head']['method'] == 'POST'){
return $this->checkPost($request);
}else{
return self::ST_FINISH;
} 
}
/**
* 檢查請求頭
* @param unknown $fd
* @param unknown $data
* @return boolean|array
*/
public function checkHeader($fd, $data){
//新的請求
if(!isset($this->requsts[$fd])){
//http頭結束符
$ret = strpos($data,self::HTTP_EOF);
if($ret === false){
return false;
}else{
$this->buffer_header[$fd] = '';
$request = array();
list($header,$request['body']) = explode(self::HTTP_EOF, $data,2); 
$request['head'] = $this->parseHeader($header); 
$this->requsts[$fd] = $request;
if($request['head'] == false){
return false;
}
}
}else{
//post 數(shù)據(jù)合并
$request = $this->requsts[$fd];
$request['body'] .= $data;
}
return $request;
}
/**
* 解析請求頭
* @param string $header
* @return array
* array(
* 'method'=>,
* 'uri'=>
* 'protocol'=>
* 'name'=>value,...
* 
* 
* 
* }
*/
public function parseHeader($header){
$request = array();
$headlines = explode("\r\n", $header);
list($request['method'],$request['uri'],$request['protocol']) = explode(' ', $headlines[0],3); 
foreach ($headlines as $k=>$line){
$line = trim($line); 
if($k && !empty($line) && strpos($line,':') !== false){
list($name,$value) = explode(':', $line,2);
$request[trim($name)] = trim($value);
}
} 
return $request;
}
/**
* 檢查post數(shù)據(jù)是否完整
* @param unknown $request
* @return string
*/
public function checkPost($request){
if(isset($request['head']['Content-Length'])){
if(intval($request['head']['Content-Length']) > self::HTTP_POST_MAXLEN){
return self::ST_ERROR;
}
if(intval($request['head']['Content-Length']) > strlen($request['body'])){
return self::ST_WAIT;
}else{
return self::ST_FINISH;
}
}
return self::ST_ERROR;
}
/**
* 解析請求
* @param unknown $request
* @return Ambigous <unknown, mixed, multitype:string >
*/
public function parseRequest($request){
$request['time'] = time();
$url_info = parse_url($request['head']['uri']);
$request['head']['path'] = $url_info['path'];
if(isset($url_info['fragment']))$request['head']['fragment'] = $url_info['fragment'];
if(isset($url_info['query'])){
parse_str($url_info['query'],$request['get']);
}
//parse post body
if($request['head']['method'] == 'POST'){
//目前只處理表單提交 
if (isset($request['head']['Content-Type']) && substr($request['head']['Content-Type'], 0, 33) == 'application/x-www-form-urlencoded'
|| isset($request['head']['X-Request-With']) && $request['head']['X-Request-With'] == 'XMLHttpRequest'){
parse_str($request['body'],$request['post']);
}
}
//parse cookies
if(!empty($request['head']['Cookie'])){
$params = array();
$blocks = explode(";", $request['head']['Cookie']);
foreach ($blocks as $b){
$_r = explode("=", $b, 2);
if(count($_r)==2){
list ($key, $value) = $_r;
$params[trim($key)] = trim($value, "\r\n \t\"");
}else{
$params[$_r[0]] = '';
}
}
$request['cookie'] = $params;
}
return $request;
}
public function parseResponse($request,$response){
if(!isset($response['head']['Date'])){
$response['head']['Date'] = gmdate("D, d M Y H:i:s T");
}
if(!isset($response['head']['Content-Type'])){
$response['head']['Content-Type'] = 'text/html;charset=utf-8';
}
if(!isset($response['head']['Content-Length'])){
$response['head']['Content-Length'] = strlen($response['body']);
}
if(!isset($response['head']['Connection'])){
if(isset($request['head']['Connection']) && strtolower($request['head']['Connection']) == 'keep-alive'){
$response['head']['Connection'] = 'keep-alive';
}else{
$response['head']['Connection'] = 'close';
} 
}
$response['head']['Server'] = CFtpServer::$software.'/'.CFtpServer::VERSION; 
$out = '';
if(isset($response['head']['Status'])){
$out .= 'HTTP/1.1 '.$response['head']['Status']."\r\n";
unset($response['head']['Status']);
}else{
$out .= "HTTP/1.1 200 OK\r\n";
}
//headers
foreach($response['head'] as $k=>$v){
$out .= $k.': '.$v."\r\n";
}
//cookies
if($_COOKIE){ 
$arr = array();
foreach ($_COOKIE as $k => $v){
$arr[] = $k.'='.$v; 
}
$out .= 'Set-Cookie: '.implode(';', $arr)."\r\n";
}
//End
$out .= "\r\n";
$out .= $response['body'];
return $out;
}
/**
* 處理請求
* @param unknown $request
* @return array
*/
public function process($request){
$path = $request['head']['path'];
$isDeny = false;
foreach ($this->config['path_deny'] as $p){
if(strpos($path, $p) === 0){
$isDeny = true;
break;
}
}
if($isDeny){
return $this->httpError(403, '服務器拒絕訪問:路徑錯誤'); 
}
if(!in_array($request['head']['method'],array('GET','POST'))){
return $this->httpError(500, '服務器拒絕訪問:錯誤的請求方法');
}
$file_ext = strtolower(trim(substr(strrchr($path, '.'), 1)));
$path = realpath(rtrim($this->config['wwwroot'],'/'). '/' . ltrim($path,'/'));
$this->log('WEB:['.$request['head']['method'].'] '.$request['head']['uri'] .' '.json_encode(isset($request['post'])?$request['post']:array()));
$response = array();
if($file_ext == 'php'){
if(is_file($path)){
//設置全局變量 
if(isset($request['get']))$_GET = $request['get'];
if(isset($request['post']))$_POST = $request['post'];
if(isset($request['cookie']))$_COOKIE = $request['cookie'];
$_REQUEST = array_merge($_GET,$_POST, $_COOKIE); 
foreach ($request['head'] as $key => $value){
$_key = 'HTTP_'.strtoupper(str_replace('-', '_', $key));
$_SERVER[$_key] = $value;
}
$_SERVER['REMOTE_ADDR'] = $request['remote_ip'];
$_SERVER['REQUEST_URI'] = $request['head']['uri']; 
//進行http auth
if(isset($_GET['c']) && strtolower($_GET['c']) != 'site'){
if(isset($request['head']['Authorization'])){
$user = new User();
if($user->checkUserBasicAuth($request['head']['Authorization'])){
$response['head']['Status'] = self::$HTTP_HEADERS[200];
goto process;
}
}
$response['head']['Status'] = self::$HTTP_HEADERS[401];
$response['head']['WWW-Authenticate'] = 'Basic realm="Real-Data-FTP"'; 
$_GET['c'] = 'Site';
$_GET['a'] = 'Unauthorized'; 
}
process: 
ob_start(); 
try{
include $path; 
$response['body'] = ob_get_contents();
$response['head']['Content-Type'] = APP::$content_type; 
}catch (Exception $e){
$response = $this->httpError(500, $e->getMessage());
}
ob_end_clean();
}else{
$response = $this->httpError(404, '頁面不存在');
}
}else{
//處理靜態(tài)文件
if(is_file($path)){
$response['head']['Content-Type'] = isset(self::$MIME_TYPES[$file_ext]) ? self::$MIME_TYPES[$file_ext]:"application/octet-stream";
//使用緩存
if(!isset($request['head']['If-Modified-Since'])){
$fstat = stat($path);
$expire = 2592000;//30 days
$response['head']['Status'] = self::$HTTP_HEADERS[200];
$response['head']['Cache-Control'] = "max-age={$expire}";
$response['head']['Pragma'] = "max-age={$expire}";
$response['head']['Last-Modified'] = date(self::DATE_FORMAT_HTTP, $fstat['mtime']);
$response['head']['Expires'] = "max-age={$expire}";
$response['body'] = file_get_contents($path);
}else{
$response['head']['Status'] = self::$HTTP_HEADERS[304];
$response['body'] = '';
} 
}else{
$response = $this->httpError(404, '頁面不存在');
} 
}
return $response;
}
public function httpError($code, $content){
$response = array();
$version = CFtpServer::$software.'/'.CFtpServer::VERSION; 
$response['head']['Content-Type'] = 'text/html;charset=utf-8';
$response['head']['Status'] = self::$HTTP_HEADERS[$code];
$response['body'] = <<<html
<!DOCTYPE html>
<html lang="zh-CN">
<head>
<meta charset="utf-8"> 
<title>FTP后臺管理 </title>
</head>
<body>
<p>{$content}</p>
<div style="text-align:center">
<hr>
{$version} Copyright &copy; 2015 by <a target='_new' >Real Data</a> All Rights Reserved.
</div>
</body>
</html>
html;
return $response;
}
static $HTTP_HEADERS = array(
100 => "100 Continue",
101 => "101 Switching Protocols",
200 => "200 OK",
201 => "201 Created",
204 => "204 No Content",
206 => "206 Partial Content",
300 => "300 Multiple Choices",
301 => "301 Moved Permanently",
302 => "302 Found",
303 => "303 See Other",
304 => "304 Not Modified",
307 => "307 Temporary Redirect",
400 => "400 Bad Request",
401 => "401 Unauthorized",
403 => "403 Forbidden",
404 => "404 Not Found",
405 => "405 Method Not Allowed",
406 => "406 Not Acceptable",
408 => "408 Request Timeout",
410 => "410 Gone",
413 => "413 Request Entity Too Large",
414 => "414 Request URI Too Long",
415 => "415 Unsupported Media Type",
416 => "416 Requested Range Not Satisfiable",
417 => "417 Expectation Failed",
500 => "500 Internal Server Error",
501 => "501 Method Not Implemented",
503 => "503 Service Unavailable",
506 => "506 Variant Also Negotiates",
);
static $MIME_TYPES = array( 
'jpg' => 'image/jpeg',
'bmp' => 'image/bmp',
'ico' => 'image/x-icon',
'gif' => 'image/gif',
'png' => 'image/png' ,
'bin' => 'application/octet-stream',
'js' => 'application/javascript',
'css' => 'text/css' ,
'html' => 'text/html' ,
'xml' => 'text/xml',
'tar' => 'application/x-tar' ,
'ppt' => 'application/vnd.ms-powerpoint',
'pdf' => 'application/pdf' ,
'svg' => ' image/svg+xml',
'woff' => 'application/x-font-woff',
'woff2' => 'application/x-font-woff', 
); 
} 

4.FTP主類

  有了前面類,就可以在ftp進行引用了。使用ssl時,請注意進行防火墻passive 端口范圍的nat配置?!?br />

defined('DEBUG_ON') or define('DEBUG_ON', false);
//主目錄
defined('BASE_PATH') or define('BASE_PATH', __DIR__);
require_once BASE_PATH.'/inc/User.php';
require_once BASE_PATH.'/inc/ShareMemory.php';
require_once BASE_PATH.'/web/CWebServer.php';
require_once BASE_PATH.'/inc/CSmtp.php';
class CFtpServer{
//軟件版本
const VERSION = '2.0'; 
const EOF = "\r\n"; 
public static $software "FTP-Server";
private static $server_mode = SWOOLE_PROCESS; 
private static $pid_file;
private static $log_file; 
//待寫入文件的日志隊列(緩沖區(qū))
private $queue = array();
private $pasv_port_range = array(55000,60000);
public $host = '0.0.0.0';
public $port = 21;
public $setting = array();
//最大連接數(shù)
public $max_connection = 50; 
//web管理端口
public $manager_port = 8080;
//tls
public $ftps_port = 990;
/**
* @var swoole_server
*/
protected $server;
protected $connection = array();
protected $session = array();
protected $user;//用戶類,復制驗證與權限
//共享內(nèi)存類
protected $shm;//ShareMemory
/**
* 
* @var embedded http server
*/
protected $webserver;
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ 靜態(tài)方法
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++*/
public static function setPidFile($pid_file){
self::$pid_file = $pid_file;
}
/**
* 服務啟動控制方法
*/
public static function start($startFunc){
if(empty(self::$pid_file)){
exit("Require pid file.\n"); 
}
if(!extension_loaded('posix')){ 
exit("Require extension `posix`.\n"); 
}
if(!extension_loaded('swoole')){ 
exit("Require extension `swoole`.\n"); 
}
if(!extension_loaded('shmop')){
exit("Require extension `shmop`.\n");
}
if(!extension_loaded('openssl')){
exit("Require extension `openssl`.\n");
}
$pid_file = self::$pid_file;
$server_pid = 0;
if(is_file($pid_file)){
$server_pid = file_get_contents($pid_file);
}
global $argv;
if(empty($argv[1])){
goto usage;
}elseif($argv[1] == 'reload'){
if (empty($server_pid)){
exit("FtpServer is not running\n");
}
posix_kill($server_pid, SIGUSR1);
exit;
}elseif ($argv[1] == 'stop'){
if (empty($server_pid)){
exit("FtpServer is not running\n");
}
posix_kill($server_pid, SIGTERM);
exit;
}elseif ($argv[1] == 'start'){
//已存在ServerPID,并且進程存在
if (!empty($server_pid) and posix_kill($server_pid,(int) 0)){
exit("FtpServer is already running.\n");
}
//啟動服務器
$startFunc(); 
}else{
usage:
exit("Usage: php {$argv[0]} start|stop|reload\n");
}
}
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ 方法
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++*/
public function __construct($host,$port){
$this->user = new User();
$this->shm = new ShareMemory();
$this->shm->write(array());
$flag = SWOOLE_SOCK_TCP;
$this->server = new swoole_server($host,$port,self::$server_mode,$flag);
$this->host = $host;
$this->port = $port;
$this->setting = array(
'backlog' => 128, 
'dispatch_mode' => 2,
); 
}
public function daemonize(){
$this->setting['daemonize'] = 1; 
}
public function getConnectionInfo($fd){
return $this->server->connection_info($fd); 
}
/**
* 啟動服務進程
* @param array $setting
* @throws Exception
*/ 
public function run($setting = array()){
$this->setting = array_merge($this->setting,$setting); 
//不使用swoole的默認日志
if(isset($this->setting['log_file'])){
self::$log_file = $this->setting['log_file'];
unset($this->setting['log_file']);
} 
if(isset($this->setting['max_connection'])){
$this->max_connection = $this->setting['max_connection'];
unset($this->setting['max_connection']);
}
if(isset($this->setting['manager_port'])){
$this->manager_port = $this->setting['manager_port'];
unset($this->setting['manager_port']);
}
if(isset($this->setting['ftps_port'])){
$this->ftps_port = $this->setting['ftps_port'];
unset($this->setting['ftps_port']);
}
if(isset($this->setting['passive_port_range'])){
$this->pasv_port_range = $this->setting['passive_port_range'];
unset($this->setting['passive_port_range']);
} 
$this->server->set($this->setting);
$version = explode('.', SWOOLE_VERSION);
if($version[0] == 1 && $version[1] < 7 && $version[2] <20){
throw new Exception('Swoole version require 1.7.20 +.');
}
//事件綁定
$this->server->on('start',array($this,'onMasterStart'));
$this->server->on('shutdown',array($this,'onMasterStop'));
$this->server->on('ManagerStart',array($this,'onManagerStart'));
$this->server->on('ManagerStop',array($this,'onManagerStop'));
$this->server->on('WorkerStart',array($this,'onWorkerStart'));
$this->server->on('WorkerStop',array($this,'onWorkerStop'));
$this->server->on('WorkerError',array($this,'onWorkerError'));
$this->server->on('Connect',array($this,'onConnect'));
$this->server->on('Receive',array($this,'onReceive'));
$this->server->on('Close',array($this,'onClose'));
//管理端口
$this->server->addlistener($this->host,$this->manager_port,SWOOLE_SOCK_TCP);
//tls
$this->server->addlistener($this->host,$this->ftps_port,SWOOLE_SOCK_TCP | SWOOLE_SSL);
$this->server->start();
}
public function log($msg,$level = 'debug',$flush = false){ 
if(DEBUG_ON){
$log = date('Y-m-d H:i:s').' ['.$level."]\t" .$msg."\n";
if(!empty(self::$log_file)){
$debug_file = dirname(self::$log_file).'/debug.log'; 
file_put_contents($debug_file, $log,FILE_APPEND);
if(filesize($debug_file) > 10485760){//10M
unlink($debug_file);
}
}
echo $log; 
}
if($level != 'debug'){
//日志記錄 
$this->queue[] = date('Y-m-d H:i:s')."\t[".$level."]\t".$msg; 
} 
if(count($this->queue)>10 && !empty(self::$log_file) || $flush){
if (filesize(self::$log_file) > 209715200){ //200M 
rename(self::$log_file,self::$log_file.'.'.date('His'));
}
$logs = '';
foreach ($this->queue as $q){
$logs .= $q."\n";
}
file_put_contents(self::$log_file, $logs,FILE_APPEND);
$this->queue = array();
} 
}
public function shutdown(){
return $this->server->shutdown();
}
public function close($fd){
return $this->server->close($fd);
}
public function send($fd,$data){
$data = strtr($data,array("\n" => "", "\0" => "", "\r" => ""));
$this->log("[-->]\t" . $data);
return $this->server->send($fd,$data.self::EOF);
}
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ 事件回調
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++*/
public function onMasterStart($serv){
global $argv;
swoole_set_process_name('php '.$argv[0].': master -host='.$this->host.' -port='.$this->port.'/'.$this->manager_port);
if(!empty($this->setting['pid_file'])){
file_put_contents(self::$pid_file, $serv->master_pid);
}
$this->log('Master started.');
}
public function onMasterStop($serv){
if (!empty($this->setting['pid_file'])){
unlink(self::$pid_file);
}
$this->shm->delete();
$this->log('Master stop.');
}
public function onManagerStart($serv){
global $argv;
swoole_set_process_name('php '.$argv[0].': manager');
$this->log('Manager started.');
}
public function onManagerStop($serv){
$this->log('Manager stop.');
}
public function onWorkerStart($serv,$worker_id){
global $argv;
if($worker_id >= $serv->setting['worker_num']) {
swoole_set_process_name("php {$argv[0]}: worker [task]");
} else {
swoole_set_process_name("php {$argv[0]}: worker [{$worker_id}]");
}
$this->log("Worker {$worker_id} started.");
}
public function onWorkerStop($serv,$worker_id){
$this->log("Worker {$worker_id} stop.");
}
public function onWorkerError($serv,$worker_id,$worker_pid,$exit_code){
$this->log("Worker {$worker_id} error:{$exit_code}.");
}
public function onConnect($serv,$fd,$from_id){
$info = $this->getConnectionInfo($fd);
if($info['server_port'] == $this->manager_port){
//web請求
$this->webserver = new CWebServer();
}else{
$this->send($fd, "220---------- Welcome to " . self::$software . " ----------");
$this->send($fd, "220-Local time is now " . date("H:i"));
$this->send($fd, "220 This is a private system - No anonymous login");
if(count($this->server->connections) <= $this->max_connection){
if($info['server_port'] == $this->port && isset($this->setting['force_ssl']) && $this->setting['force_ssl']){
//如果啟用強制ssl 
$this->send($fd, "421 Require implicit FTP over tls, closing control connection.");
$this->close($fd);
return ;
}
$this->connection[$fd] = array();
$this->session = array();
$this->queue = array(); 
}else{ 
$this->send($fd, "421 Too many connections, closing control connection.");
$this->close($fd);
}
}
}
public function onReceive($serv,$fd,$from_id,$recv_data){
$info = $this->getConnectionInfo($fd);
if($info['server_port'] == $this->manager_port){
//web請求
$this->webserver->onReceive($this->server, $fd, $recv_data);
}else{
$read = trim($recv_data);
$this->log("[<--]\t" . $read);
$cmd = explode(" ", $read); 
$func = 'cmd_'.strtoupper($cmd[0]);
$data = trim(str_replace($cmd[0], '', $read));
if (!method_exists($this, $func)){
$this->send($fd, "500 Unknown Command");
return;
}
if (empty($this->connection[$fd]['login'])){
switch($cmd[0]){
case 'TYPE':
case 'USER':
case 'PASS':
case 'QUIT':
case 'AUTH':
case 'PBSZ':
break;
default:
$this->send($fd,"530 You aren't logged in");
return;
}
}
$this->$func($fd,$data);
}
} 
public function onClose($serv,$fd,$from_id){
//在線用戶 
$shm_data = $this->shm->read();
if($shm_data !== false){
if(isset($shm_data['online'])){
$list = array();
foreach($shm_data['online'] as $u => $info){ 
if(!preg_match('/\.*-'.$fd.'$/',$u,$m))
$list[$u] = $info;
}
$shm_data['online'] = $list;
$this->shm->write($shm_data); 
} 
}
$this->log('Socket '.$fd.' close. Flush the logs.','debug',true);
}
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ 工具函數(shù)
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++*/ 
/**
* 獲取用戶名
* @param $fd
*/
public function getUser($fd){
return isset($this->connection[$fd]['user'])?$this->connection[$fd]['user']:'';
}
/**
* 獲取文件全路徑
* @param $user
* @param $file
* @return string|boolean
*/
public function getFile($user, $file){
$file = $this->fillDirName($user, $file); 
if (is_file($file)){
return realpath($file);
}else{
return false;
}
}
/**
* 遍歷目錄
* @param $rdir
* @param $showHidden
* @param $format list/mlsd
* @return string
* 
* list 使用local時間
* mlsd 使用gmt時間
*/
public function getFileList($user, $rdir, $showHidden = false, $format = 'list'){
$filelist = '';
if($format == 'mlsd'){
$stats = stat($rdir);
$filelist.= 'Type=cdir;Modify='.gmdate('YmdHis',$stats['mtime']).';UNIX.mode=d'.$this->mode2char($stats['mode']).'; '.$this->getUserDir($user)."\r\n";
}
if ($handle = opendir($rdir)){
$isListable = $this->user->isFolderListable($user, $rdir);
while (false !== ($file = readdir($handle))){
if ($file == '.' or $file == '..'){
continue;
}
if ($file{0} == "." and !$showHidden){
continue;
}
//如果當前目錄$rdir不允許列出,則判斷當前目錄下的目錄是否配置為可以列出 
if(!$isListable){ 
$dir = $rdir . $file;
if(is_dir($dir)){
$dir = $this->joinPath($dir, '/');
if($this->user->isFolderListable($user, $dir)){ 
goto listFolder;
}
}
continue;
} 
listFolder: 
$stats = stat($rdir . $file);
if (is_dir($rdir . "/" . $file)) $mode = "d"; else $mode = "-";
$mode .= $this->mode2char($stats['mode']);
if($format == 'mlsd'){
if($mode[0] == 'd'){
$filelist.= 'Type=dir;Modify='.gmdate('YmdHis',$stats['mtime']).';UNIX.mode='.$mode.'; '.$file."\r\n";
}else{
$filelist.= 'Type=file;Size='.$stats['size'].';Modify='.gmdate('YmdHis',$stats['mtime']).';UNIX.mode='.$mode.'; '.$file."\r\n";
}
}else{
$uidfill = "";
for ($i = strlen($stats['uid']); $i < 5; $i++) $uidfill .= " ";
$gidfill = "";
for ($i = strlen($stats['gid']); $i < 5; $i++) $gidfill .= " ";
$sizefill = "";
for ($i = strlen($stats['size']); $i < 11; $i++) $sizefill .= " ";
$nlinkfill = "";
for ($i = strlen($stats['nlink']); $i < 5; $i++) $nlinkfill .= " ";
$mtime = date("M d H:i", $stats['mtime']);
$filelist .= $mode . $nlinkfill . $stats['nlink'] . " " . $stats['uid'] . $uidfill . $stats['gid'] . $gidfill . $sizefill . $stats['size'] . " " . $mtime . " " . $file . "\r\n";
}
}
closedir($handle);
}
return $filelist;
}
/**
* 將文件的全新從數(shù)字轉換為字符串
* @param int $int
*/
public function mode2char($int){
$mode = '';
$moded = sprintf("%o", ($int & 000777));
$mode1 = substr($moded, 0, 1);
$mode2 = substr($moded, 1, 1);
$mode3 = substr($moded, 2, 1);
switch ($mode1) {
case "0":
$mode .= "---";
break;
case "1":
$mode .= "--x";
break;
case "2":
$mode .= "-w-";
break;
case "3":
$mode .= "-wx";
break;
case "4":
$mode .= "r--";
break;
case "5":
$mode .= "r-x";
break;
case "6":
$mode .= "rw-";
break;
case "7":
$mode .= "rwx";
break;
}
switch ($mode2) {
case "0":
$mode .= "---";
break;
case "1":
$mode .= "--x";
break;
case "2":
$mode .= "-w-";
break;
case "3":
$mode .= "-wx";
break;
case "4":
$mode .= "r--";
break;
case "5":
$mode .= "r-x";
break;
case "6":
$mode .= "rw-";
break;
case "7":
$mode .= "rwx";
break;
}
switch ($mode3) {
case "0":
$mode .= "---";
break;
case "1":
$mode .= "--x";
break;
case "2":
$mode .= "-w-";
break;
case "3":
$mode .= "-wx";
break;
case "4":
$mode .= "r--";
break;
case "5":
$mode .= "r-x";
break;
case "6":
$mode .= "rw-";
break;
case "7":
$mode .= "rwx";
break;
}
return $mode;
}
/**
* 設置用戶當前的路徑 
* @param $user
* @param $pwd
*/
public function setUserDir($user, $cdir){
$old_dir = $this->session[$user]['pwd'];
if ($old_dir == $cdir){
return $cdir;
} 
if($cdir[0] != '/')
$cdir = $this->joinPath($old_dir,$cdir); 
$this->session[$user]['pwd'] = $cdir;
$abs_dir = realpath($this->getAbsDir($user));
if (!$abs_dir){
$this->session[$user]['pwd'] = $old_dir;
return false;
}
$this->session[$user]['pwd'] = $this->joinPath('/',substr($abs_dir, strlen($this->session[$user]['home'])));
$this->session[$user]['pwd'] = $this->joinPath($this->session[$user]['pwd'],'/');
$this->log("CHDIR: $old_dir -> $cdir");
return $this->session[$user]['pwd'];
}
/**
* 獲取全路徑
* @param $user
* @param $file
* @return string
*/
public function fillDirName($user, $file){ 
if (substr($file, 0, 1) != "/"){
$file = '/'.$file;
$file = $this->joinPath($this->getUserDir( $user), $file);
} 
$file = $this->joinPath($this->session[$user]['home'],$file);
return $file;
}
/**
* 獲取用戶路徑
* @param unknown $user
*/
public function getUserDir($user){
return $this->session[$user]['pwd'];
}
/**
* 獲取用戶的當前文件系統(tǒng)絕對路徑,非chroot路徑
* @param $user
* @return string
*/
public function getAbsDir($user){
$rdir = $this->joinPath($this->session[$user]['home'],$this->session[$user]['pwd']);
return $rdir;
}
/**
* 路徑連接
* @param string $path1
* @param string $path2
* @return string
*/
public function joinPath($path1,$path2){ 
$path1 = rtrim($path1,'/');
$path2 = trim($path2,'/');
return $path1.'/'.$path2;
}
/**
* IP判斷
* @param string $ip
* @return boolean
*/
public function isIPAddress($ip){
if (!is_numeric($ip[0]) || $ip[0] < 1 || $ip[0] > 254) {
return false;
} elseif (!is_numeric($ip[1]) || $ip[1] < 0 || $ip[1] > 254) {
return false;
} elseif (!is_numeric($ip[2]) || $ip[2] < 0 || $ip[2] > 254) {
return false;
} elseif (!is_numeric($ip[3]) || $ip[3] < 1 || $ip[3] > 254) {
return false;
} elseif (!is_numeric($ip[4]) || $ip[4] < 1 || $ip[4] > 500) {
return false;
} elseif (!is_numeric($ip[5]) || $ip[5] < 1 || $ip[5] > 500) {
return false;
} else {
return true;
}
}
/**
* 獲取pasv端口
* @return number
*/
public function getPasvPort(){
$min = is_int($this->pasv_port_range[0])?$this->pasv_port_range[0]:55000;
$max = is_int($this->pasv_port_range[1])?$this->pasv_port_range[1]:60000;
$max = $max <= 65535 ? $max : 65535;
$loop = 0;
$port = 0;
while($loop < 10){
$port = mt_rand($min, $max);
if($this->isAvailablePasvPort($port)){ 
break;
}
$loop++;
} 
return $port;
}
public function pushPasvPort($port){
$shm_data = $this->shm->read();
if($shm_data !== false){
if(isset($shm_data['pasv_port'])){
array_push($shm_data['pasv_port'], $port);
}else{
$shm_data['pasv_port'] = array($port);
}
$this->shm->write($shm_data);
$this->log('Push pasv port: '.implode(',', $shm_data['pasv_port']));
return true;
}
return false;
}
public function popPasvPort($port){
$shm_data = $this->shm->read();
if($shm_data !== false){
if(isset($shm_data['pasv_port'])){
$tmp = array();
foreach ($shm_data['pasv_port'] as $p){
if($p != $port){
$tmp[] = $p;
}
}
$shm_data['pasv_port'] = $tmp;
}
$this->shm->write($shm_data);
$this->log('Pop pasv port: '.implode(',', $shm_data['pasv_port']));
return true;
}
return false;
}
public function isAvailablePasvPort($port){
$shm_data = $this->shm->read();
if($shm_data !== false){
if(isset($shm_data['pasv_port'])){
return !in_array($port, $shm_data['pasv_port']);
}
return true;
}
return false;
}
/**
* 獲取當前數(shù)據(jù)鏈接tcp個數(shù)
*/
public function getDataConnections(){
$shm_data = $this->shm->read();
if($shm_data !== false){
if(isset($shm_data['pasv_port'])){
return count($shm_data['pasv_port']);
} 
}
return 0;
} 
/**
* 關閉數(shù)據(jù)傳輸socket
* @param $user
* @return bool
*/
public function closeUserSock($user){
$peer = stream_socket_get_name($this->session[$user]['sock'], false);
list($ip,$port) = explode(':', $peer);
//釋放端口占用
$this->popPasvPort($port);
fclose($this->session[$user]['sock']);
$this->session[$user]['sock'] = 0;
return true;
}
/**
* @param $user
* @return resource
*/
public function getUserSock($user){
//被動模式
if ($this->session[$user]['pasv'] == true){
if (empty($this->session[$user]['sock'])){
$addr = stream_socket_get_name($this->session[$user]['serv_sock'], false);
list($ip, $port) = explode(':', $addr);
$sock = stream_socket_accept($this->session[$user]['serv_sock'], 5);
if ($sock){
$peer = stream_socket_get_name($sock, true);
$this->log("Accept: success client is $peer.");
$this->session[$user]['sock'] = $sock;
//關閉server socket
fclose($this->session[$user]['serv_sock']);
}else{
$this->log("Accept: failed.");
//釋放端口
$this->popPasvPort($port);
return false;
}
}
}
return $this->session[$user]['sock'];
}
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ FTP Command
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++*/
//==================
//RFC959
//==================
/**
* 登錄用戶名
* @param $fd
* @param $data
*/
public function cmd_USER($fd, $data){
if (preg_match("/^([a-z0-9.@]+)$/", $data)){
$user = strtolower($data);
$this->connection[$fd]['user'] = $user; 
$this->send($fd, "331 User $user OK. Password required");
}else{
$this->send($fd, "530 Login authentication failed");
}
}
/**
* 登錄密碼
* @param $fd
* @param $data
*/
public function cmd_PASS($fd, $data){
$user = $this->connection[$fd]['user'];
$pass = $data;
$info = $this->getConnectionInfo($fd);
$ip = $info['remote_ip'];
//判斷登陸失敗次數(shù)
if($this->user->isAttemptLimit($this->shm, $user, $ip)){
$this->send($fd, "530 Login authentication failed: Too many login attempts. Blocked in 10 minutes.");
return;
} 
if ($this->user->checkUser($user, $pass, $ip)){
$dir = "/";
$this->session[$user]['pwd'] = $dir;
//ftp根目錄 
$this->session[$user]['home'] = $this->user->getHomeDir($user);
if(empty($this->session[$user]['home']) || !is_dir($this->session[$user]['home'])){
$this->send($fd, "530 Login authentication failed: `home` path error.");
}else{
$this->connection[$fd]['login'] = true;
//在線用戶
$shm_data = $this->user->addOnline($this->shm, $this->server, $user, $fd, $ip);
$this->log('SHM: '.json_encode($shm_data) );
$this->send($fd, "230 OK. Current restricted directory is " . $dir); 
$this->log('User '.$user .' has login successfully! IP: '.$ip,'warn');
}
}else{
$this->user->addAttempt($this->shm, $user, $ip);
$this->log('User '.$user .' login fail! IP: '.$ip,'warn');
$this->send($fd, "530 Login authentication failed: check your pass or ip allow rules.");
}
}
/**
* 更改當前目錄
* @param $fd
* @param $data
*/
public function cmd_CWD($fd, $data){
$user = $this->getUser($fd);
if (($dir = $this->setUserDir($user, $data)) != false){
$this->send($fd, "250 OK. Current directory is " . $dir);
}else{
$this->send($fd, "550 Can't change directory to " . $data . ": No such file or directory");
}
}
/**
* 返回上級目錄
* @param $fd
* @param $data
*/
public function cmd_CDUP($fd, $data){
$data = '..';
$this->cmd_CWD($fd, $data);
}
/**
* 退出服務器
* @param $fd
* @param $data
*/
public function cmd_QUIT($fd, $data){
$this->send($fd,"221 Goodbye.");
unset($this->connection[$fd]);
}
/**
* 獲取當前目錄
* @param $fd
* @param $data
*/
public function cmd_PWD($fd, $data){
$user = $this->getUser($fd);
$this->send($fd, "257 \"" . $this->getUserDir($user) . "\" is your current location");
}
/**
* 下載文件
* @param $fd
* @param $data
*/
public function cmd_RETR($fd, $data){
$user = $this->getUser($fd);
$ftpsock = $this->getUserSock($user);
if (!$ftpsock){
$this->send($fd, "425 Connection Error");
return;
}
if (($file = $this->getFile($user, $data)) != false){
if($this->user->isReadable($user, $file)){
$this->send($fd, "150 Connecting to client");
if ($fp = fopen($file, "rb")){
//斷點續(xù)傳
if(isset($this->session[$user]['rest_offset'])){
if(!fseek($fp, $this->session[$user]['rest_offset'])){
$this->log("RETR at offset ".ftell($fp));
}else{
$this->log("RETR at offset ".ftell($fp).' fail.');
}
unset($this->session[$user]['rest_offset']);
} 
while (!feof($fp)){ 
$cont = fread($fp, 8192); 
if (!fwrite($ftpsock, $cont)) break; 
}
if (fclose($fp) and $this->closeUserSock($user)){
$this->send($fd, "226 File successfully transferred");
$this->log($user."\tGET:".$file,'info');
}else{
$this->send($fd, "550 Error during file-transfer");
}
}else{
$this->send($fd, "550 Can't open " . $data . ": Permission denied");
}
}else{
$this->send($fd, "550 You're unauthorized: Permission denied");
}
}else{
$this->send($fd, "550 Can't open " . $data . ": No such file or directory");
}
}
/**
* 上傳文件
* @param $fd
* @param $data
*/
public function cmd_STOR($fd, $data){
$user = $this->getUser($fd);
$ftpsock = $this->getUserSock($user);
if (!$ftpsock){
$this->send($fd, "425 Connection Error");
return;
}
$file = $this->fillDirName($user, $data);
$isExist = false;
if(file_exists($file))$isExist = true;
if((!$isExist && $this->user->isWritable($user, $file)) ||
($isExist && $this->user->isAppendable($user, $file))){
if($isExist){
$fp = fopen($file, "rb+");
$this->log("OPEN for STOR.");
}else{
$fp = fopen($file, 'wb');
$this->log("CREATE for STOR.");
}
if (!$fp){
$this->send($fd, "553 Can't open that file: Permission denied");
}else{
//斷點續(xù)傳,需要Append權限
if(isset($this->session[$user]['rest_offset'])){
if(!fseek($fp, $this->session[$user]['rest_offset'])){
$this->log("STOR at offset ".ftell($fp));
}else{
$this->log("STOR at offset ".ftell($fp).' fail.');
}
unset($this->session[$user]['rest_offset']);
}
$this->send($fd, "150 Connecting to client");
while (!feof($ftpsock)){
$cont = fread($ftpsock, 8192);
if (!$cont) break;
if (!fwrite($fp, $cont)) break;
}
touch($file);//設定文件的訪問和修改時間
if (fclose($fp) and $this->closeUserSock($user)){
$this->send($fd, "226 File successfully transferred");
$this->log($user."\tPUT: $file",'info');
}else{
$this->send($fd, "550 Error during file-transfer");
}
}
}else{
$this->send($fd, "550 You're unauthorized: Permission denied");
$this->closeUserSock($user);
}
}
/**
* 文件追加
* @param $fd
* @param $data
*/
public function cmd_APPE($fd,$data){
$user = $this->getUser($fd);
$ftpsock = $this->getUserSock($user);
if (!$ftpsock){
$this->send($fd, "425 Connection Error");
return;
}
$file = $this->fillDirName($user, $data);
$isExist = false;
if(file_exists($file))$isExist = true;
if((!$isExist && $this->user->isWritable($user, $file)) ||
($isExist && $this->user->isAppendable($user, $file))){
$fp = fopen($file, "rb+");
if (!$fp){
$this->send($fd, "553 Can't open that file: Permission denied");
}else{
//斷點續(xù)傳,需要Append權限
if(isset($this->session[$user]['rest_offset'])){
if(!fseek($fp, $this->session[$user]['rest_offset'])){
$this->log("APPE at offset ".ftell($fp));
}else{
$this->log("APPE at offset ".ftell($fp).' fail.');
}
unset($this->session[$user]['rest_offset']);
}
$this->send($fd, "150 Connecting to client");
while (!feof($ftpsock)){
$cont = fread($ftpsock, 8192);
if (!$cont) break;
if (!fwrite($fp, $cont)) break;
}
touch($file);//設定文件的訪問和修改時間
if (fclose($fp) and $this->closeUserSock($user)){
$this->send($fd, "226 File successfully transferred");
$this->log($user."\tAPPE: $file",'info');
}else{
$this->send($fd, "550 Error during file-transfer");
}
}
}else{
$this->send($fd, "550 You're unauthorized: Permission denied");
$this->closeUserSock($user);
}
}
/**
* 文件重命名,源文件
* @param $fd
* @param $data
*/
public function cmd_RNFR($fd, $data){
$user = $this->getUser($fd);
$file = $this->fillDirName($user, $data);
if (file_exists($file) || is_dir($file)){
$this->session[$user]['rename'] = $file;
$this->send($fd, "350 RNFR accepted - file exists, ready for destination"); 
}else{
$this->send($fd, "550 Sorry, but that '$data' doesn't exist");
}
}
/**
* 文件重命名,目標文件
* @param $fd
* @param $data
*/
public function cmd_RNTO($fd, $data){
$user = $this->getUser($fd);
$old_file = $this->session[$user]['rename'];
$new_file = $this->fillDirName($user, $data);
$isDir = false;
if(is_dir($old_file)){
$isDir = true;
$old_file = $this->joinPath($old_file, '/');
}
if((!$isDir && $this->user->isRenamable($user, $old_file)) || 
($isDir && $this->user->isFolderRenamable($user, $old_file))){
if (empty($old_file) or !is_dir(dirname($new_file))){
$this->send($fd, "451 Rename/move failure: No such file or directory");
}elseif (rename($old_file, $new_file)){
$this->send($fd, "250 File successfully renamed or moved");
$this->log($user."\tRENAME: $old_file to $new_file",'warn');
}else{
$this->send($fd, "451 Rename/move failure: Operation not permitted");
}
}else{
$this->send($fd, "550 You're unauthorized: Permission denied");
}
unset($this->session[$user]['rename']);
}
/**
* 刪除文件
* @param $fd
* @param $data
*/
public function cmd_DELE($fd, $data){
$user = $this->getUser($fd);
$file = $this->fillDirName($user, $data);
if($this->user->isDeletable($user, $file)){
if (!file_exists($file)){
$this->send($fd, "550 Could not delete " . $data . ": No such file or directory");
}
elseif (unlink($file)){
$this->send($fd, "250 Deleted " . $data);
$this->log($user."\tDEL: $file",'warn');
}else{
$this->send($fd, "550 Could not delete " . $data . ": Permission denied");
}
}else{
$this->send($fd, "550 You're unauthorized: Permission denied");
}
}
/**
* 創(chuàng)建目錄
* @param $fd
* @param $data
*/
public function cmd_MKD($fd, $data){
$user = $this->getUser($fd);
$path = '';
if($data[0] == '/'){
$path = $this->joinPath($this->session[$user]['home'],$data);
}else{
$path = $this->joinPath($this->getAbsDir($user),$data);
}
$path = $this->joinPath($path, '/'); 
if($this->user->isFolderCreatable($user, $path)){
if (!is_dir(dirname($path))){
$this->send($fd, "550 Can't create directory: No such file or directory");
}elseif(file_exists($path)){
$this->send($fd, "550 Can't create directory: File exists");
}else{
if (mkdir($path)){
$this->send($fd, "257 \"" . $data . "\" : The directory was successfully created");
$this->log($user."\tMKDIR: $path",'info');
}else{
$this->send($fd, "550 Can't create directory: Permission denied");
}
}
}else{
$this->send($fd, "550 You're unauthorized: Permission denied");
}
}
/**
* 刪除目錄
* @param $fd
* @param $data
*/
public function cmd_RMD($fd, $data){
$user = $this->getUser($fd);
$dir = '';
if($data[0] == '/'){
$dir = $this->joinPath($this->session[$user]['home'], $data);
}else{
$dir = $this->fillDirName($user, $data);
}
$dir = $this->joinPath($dir, '/');
if($this->user->isFolderDeletable($user, $dir)){
if (is_dir(dirname($dir)) and is_dir($dir)){
if (count(glob($dir . "/*"))){
$this->send($fd, "550 Can't remove directory: Directory not empty");
}elseif (rmdir($dir)){
$this->send($fd, "250 The directory was successfully removed");
$this->log($user."\tRMDIR: $dir",'warn');
}else{
$this->send($fd, "550 Can't remove directory: Operation not permitted");
}
}elseif (is_dir(dirname($dir)) and file_exists($dir)){
$this->send($fd, "550 Can't remove directory: Not a directory");
}else{
$this->send($fd, "550 Can't create directory: No such file or directory");
}
}else{
$this->send($fd, "550 You're unauthorized: Permission denied");
}
}
/**
* 得到服務器類型
* @param $fd
* @param $data
*/
public function cmd_SYST($fd, $data){
$this->send($fd, "215 UNIX Type: L8");
}
/**
* 權限控制
* @param $fd
* @param $data
*/
public function cmd_SITE($fd, $data){
if (substr($data, 0, 6) == "CHMOD "){
$user = $this->getUser($fd);
$chmod = explode(" ", $data, 3);
$file = $this->fillDirName($user, $chmod[2]);
if($this->user->isWritable($user, $file)){
if (chmod($file, octdec($chmod[1]))){
$this->send($fd, "200 Permissions changed on {$chmod[2]}");
$this->log($user."\tCHMOD: $file to {$chmod[1]}",'info');
}else{
$this->send($fd, "550 Could not change perms on " . $chmod[2] . ": Permission denied");
}
}else{
$this->send($fd, "550 You're unauthorized: Permission denied");
}
}else{
$this->send($fd, "500 Unknown Command");
}
} 
/**
* 更改傳輸類型
* @param $fd
* @param $data
*/
public function cmd_TYPE($fd, $data){
switch ($data){
case "A":
$type = "ASCII";
break;
case "I":
$type = "8-bit binary";
break;
}
$this->send($fd, "200 TYPE is now " . $type);
}
/**
* 遍歷目錄
* @param $fd
* @param $data
*/
public function cmd_LIST($fd, $data){
$user = $this->getUser($fd);
$ftpsock = $this->getUserSock($user);
if (!$ftpsock){
$this->send($fd, "425 Connection Error");
return;
} 
$path = $this->joinPath($this->getAbsDir($user),'/');
$this->send($fd, "150 Opening ASCII mode data connection for file list");
$filelist = $this->getFileList($user, $path, true);
fwrite($ftpsock, $filelist); 
$this->send($fd, "226 Transfer complete."); 
$this->closeUserSock($user);
}
/**
* 建立數(shù)據(jù)傳輸通
* @param $fd
* @param $data
*/
// 不使用主動模式 
// public function cmd_PORT($fd, $data){
// $user = $this->getUser($fd);
// $port = explode(",", $data);
// if (count($port) != 6){
// $this->send($fd, "501 Syntax error in IP address");
// }else{
// if (!$this->isIPAddress($port)){
// $this->send($fd, "501 Syntax error in IP address");
// return;
// }
// $ip = $port[0] . "." . $port[1] . "." . $port[2] . "." . $port[3];
// $port = hexdec(dechex($port[4]) . dechex($port[5]));
// if ($port < 1024){
// $this->send($fd, "501 Sorry, but I won't connect to ports < 1024");
// }elseif ($port > 65000){
// $this->send($fd, "501 Sorry, but I won't connect to ports > 65000");
// }else{ 
// $ftpsock = fsockopen($ip, $port); 
// if ($ftpsock){
// $this->session[$user]['sock'] = $ftpsock;
// $this->session[$user]['pasv'] = false; 
// $this->send($fd, "200 PORT command successful"); 
// }else{
// $this->send($fd, "501 Connection failed");
// }
// }
// }
// }
/**
* 被動模式 
* @param unknown $fd
* @param unknown $data
*/
public function cmd_PASV($fd, $data){
$user = $this->getUser($fd);
$ssl = false;
$pasv_port = $this->getPasvPort();
if($this->connection[$fd]['ssl'] === true){
$ssl = true;
$context = stream_context_create(); 
// local_cert must be in PEM format
stream_context_set_option($context, 'ssl', 'local_cert', $this->setting['ssl_cert_file']);
// Path to local private key file 
stream_context_set_option($context, 'ssl', 'local_pk', $this->setting['ssl_key_file']);
stream_context_set_option($context, 'ssl', 'allow_self_signed', true);
stream_context_set_option($context, 'ssl', 'verify_peer', false);
stream_context_set_option($context, 'ssl', 'verify_peer_name', false); 
stream_context_set_option($context, 'ssl', 'passphrase', '');
// Create the server socket
$sock = stream_socket_server('ssl://0.0.0.0:'.$pasv_port, $errno, $errstr, STREAM_SERVER_BIND | STREAM_SERVER_LISTEN, $context);
}else{
$sock = stream_socket_server('tcp://0.0.0.0:'.$pasv_port, $errno, $errstr, STREAM_SERVER_BIND | STREAM_SERVER_LISTEN);
}
if ($sock){
$addr = stream_socket_get_name($sock, false);
list($ip, $port) = explode(':', $addr);
$ipArr = swoole_get_local_ip();
foreach($ipArr as $nic => $addr){
$ip = $addr;
}
$this->log("ServerSock: $ip:$port");
$ip = str_replace('.', ',', $ip);
$this->send($fd, "227 Entering Passive Mode ({$ip},".(intval($port) >> 8 & 0xff).",".(intval($port) & 0xff)."). ".$port." ".($ssl?'ssl':''));
$this->session[$user]['serv_sock'] = $sock;
$this->session[$user]['pasv'] = true;
$this->pushPasvPort($port);
}else{
fclose($sock);
$this->send($fd, "500 failed to create data socket: ".$errstr);
}
}
public function cmd_NOOP($fd,$data){
$this->send($fd, "200 OK");
}
//==================
//RFC2228
//==================
public function cmd_PBSZ($fd,$data){
$this->send($fd, '200 Command okay.');
}
public function cmd_PROT($fd,$data){
if(trim($data) == 'P'){
$this->connection[$fd]['ssl'] = true;
$this->send($fd, '200 Set Private level on data connection.');
}elseif(trim($data) == 'C'){
$this->connection[$fd]['ssl'] = false;
$this->send($fd, '200 Set Clear level on data connection.');
}else{
$this->send($fd, '504 Command not implemented for that parameter.');
}
}
//==================
//RFC2389
//==================
public function cmd_FEAT($fd,$data){
$this->send($fd, '211-Features supported');
$this->send($fd, 'MDTM');
$this->send($fd, 'SIZE');
$this->send($fd, 'SITE CHMOD');
$this->send($fd, 'REST STREAM');
$this->send($fd, 'MLSD Type*;Size*;Modify*;UNIX.mode*;');
$this->send($fd, 'PBSZ');
$this->send($fd, 'PROT');
$this->send($fd, '211 End');
}
//關閉utf8對中文文件名有影響
public function cmd_OPTS($fd,$data){
$this->send($fd, '502 Command not implemented.');
}
//==================
//RFC3659
//==================
/**
* 獲取文件修改時間
* @param unknown $fd
* @param unknown $data
*/
public function cmd_MDTM($fd,$data){
$user = $this->getUser($fd);
if (($file = $this->getFile($user, $data)) != false){
$this->send($fd, '213 '.date('YmdHis.u',filemtime($file)));
}else{
$this->send($fd, '550 No file named "'.$data.'"');
}
}
/**
* 獲取文件大小
* @param $fd
* @param $data
*/
public function cmd_SIZE($fd,$data){
$user = $this->getUser($fd);
if (($file = $this->getFile($user, $data)) != false){
$this->send($fd, '213 '.filesize($file));
}else{
$this->send($fd, '550 No file named "'.$data.'"');
}
}
/**
* 獲取文件列表
* @param unknown $fd
* @param unknown $data
*/
public function cmd_MLSD($fd,$data){
$user = $this->getUser($fd);
$ftpsock = $this->getUserSock($user);
if (!$ftpsock){
$this->send($fd, "425 Connection Error");
return;
}
$path = $this->joinPath($this->getAbsDir($user),'/');
$this->send($fd, "150 Opening ASCII mode data connection for file list");
$filelist = $this->getFileList($user, $path, true,'mlsd');
fwrite($ftpsock, $filelist);
$this->send($fd, "226 Transfer complete.");
$this->closeUserSock($user);
}
/**
* 設置文件offset
* @param unknown $fd
* @param unknown $data
*/
public function cmd_REST($fd,$data){
$user = $this->getUser($fd);
$data= preg_replace('/[^0-9]/', '', $data);
if($data != ''){
$this->session[$user]['rest_offset'] = $data;
$this->send($fd, '350 Restarting at '.$data.'. Send STOR or RETR');
}else{
$this->send($fd, '500 Syntax error, offset unrecognized.');
}
}
/**
* 獲取文件hash值
* @param unknown $fd
* @param unknown $data
*/
public function cmd_HASH($fd,$data){
$user = $this->getUser($fd);
$ftpsock = $this->getUserSock($user);
if (($file = $this->getFile($user, $data)) != false){
if(is_file($file)){
$algo = 'sha512';
$this->send($fd, "200 ".hash_file($algo, $file));
}else{
$this->send($fd, "550 Can't open " . $data . ": No such file。");
} 
}else{
$this->send($fd, "550 Can't open " . $data . ": No such file。");
}
}
/**
* 控制臺命令
* @param unknown $fd
* @param unknown $data
*/
public function cmd_CONSOLE($fd,$data){
$group = $this->user->getUserProfile($this->getUser($fd));
$group = $group['group'];
if($group != 'admin'){
$this->send($fd, "550 You're unauthorized: Permission denied");
return;
} 
$data = explode('||', $data);
$cmd = strtoupper($data[0]);
switch ($cmd){ 
case 'USER-ONLINE':
$shm_data = $this->shm->read(); 
$list = array();
if($shm_data !== false){
if(isset($shm_data['online'])){
$list = $shm_data['online'];
} 
}
$this->send($fd, '200 '.json_encode($list));
break; 
//Format: user-add||{"user":"","pass":"","home":"","expired":"","active":boolean,"group":"","description":"","email":""}
case 'USER-ADD':
if(isset($data[1])){
$json = json_decode(trim($data[1]),true);
$user = isset($json['user'])?$json['user']:'';
$pass = isset($json['pass'])?$json['pass']:'';
$home = isset($json['home'])?$json['home']:'';
$expired = isset($json['expired'])?$json['expired']:'1999-01-01';
$active = isset($json['active'])?$json['active']:false;
$group = isset($json['group'])?$json['group']:'';
$description = isset($json['description'])?$json['description']:'';
$email = isset($json['email'])?$json['email']:'';
if($this->user->addUser($user,$pass,$home,$expired,$active,$group,$description,$email)){
$this->user->save();
$this->user->reload();
$this->send($fd, '200 User "'.$user.'" added.');
}else{
$this->send($fd, '550 Add fail!');
}
}else{
$this->send($fd, '500 Syntax error: USER-ADD||{"user":"","pass":"","home":"","expired":"","active":boolean,"group":"","description":""}');
}
break;
//Format: user-set-profile||{"user":"","profile":[]}
case 'USER-SET-PROFILE':
if(isset($data[1])){
$json = json_decode(trim($data[1]),true);
$user = isset($json['user'])?$json['user']:'';
$profile = isset($json['profile'])?$json['profile']:array(); 
if($this->user->setUserProfile($user, $profile)){
$this->user->save();
$this->user->reload();
$this->send($fd, '200 User "'.$user.'" profile changed.');
}else{
$this->send($fd, '550 Set profile fail!');
}
}else{
$this->send($fd, '500 Syntax error: USER-SET-PROFILE||{"user":"","profile":[]}');
} 
break; 
//Format: user-get-profile||{"user":""}
case 'USER-GET-PROFILE':
if(isset($data[1])){
$json = json_decode(trim($data[1]),true);
$user = isset($json['user'])?$json['user']:'';
$this->user->reload();
if($profile = $this->user->getUserProfile($user)){ 
$this->send($fd, '200 '.json_encode($profile));
}else{
$this->send($fd, '550 Get profile fail!');
}
}else{
$this->send($fd, '500 Syntax error: USER-GET-PROFILE||{"user":""}');
} 
break;
//Format: user-delete||{"user":""}
case 'USER-DELETE':
if(isset($data[1])){
$json = json_decode(trim($data[1]),true);
$user = isset($json['user'])?$json['user']:'';
if($this->user->delUser($user)){
$this->user->save();
$this->user->reload();
$this->send($fd, '200 User '.$user.' deleted.');
}else{
$this->send($fd, '550 Delete user fail!');
}
}else{
$this->send($fd, '500 Syntax error: USER-DELETE||{"user":""}');
}
break;
case 'USER-LIST':
$this->user->reload();
$list = $this->user->getUserList();
$this->send($fd, '200 '.json_encode($list));
break;
//Format: group-add||{"group":"","home":""}
case 'GROUP-ADD':
if(isset($data[1])){
$json = json_decode(trim($data[1]),true);
$group = isset($json['group'])?$json['group']:''; 
$home = isset($json['home'])?$json['home']:''; 
if($this->user->addGroup($group, $home)){
$this->user->save();
$this->user->reload();
$this->send($fd, '200 Group "'.$group.'" added.');
}else{
$this->send($fd, '550 Add group fail!');
}
}else{
$this->send($fd, '500 Syntax error: GROUP-ADD||{"group":"","home":""}');
}
break;
//Format: group-set-profile||{"group":"","profile":[]}
case 'GROUP-SET-PROFILE':
if(isset($data[1])){
$json = json_decode(trim($data[1]),true);
$group = isset($json['group'])?$json['group']:'';
$profile = isset($json['profile'])?$json['profile']:array();
if($this->user->setGroupProfile($group, $profile)){
$this->user->save();
$this->user->reload();
$this->send($fd, '200 Group "'.$group.'" profile changed.');
}else{
$this->send($fd, '550 Set profile fail!');
}
}else{
$this->send($fd, '500 Syntax error: GROUP-SET-PROFILE||{"group":"","profile":[]}');
}
break;
//Format: group-get-profile||{"group":""}
case 'GROUP-GET-PROFILE':
if(isset($data[1])){
$json = json_decode(trim($data[1]),true);
$group = isset($json['group'])?$json['group']:'';
$this->user->reload();
if($profile = $this->user->getGroupProfile($group)){
$this->send($fd, '200 '.json_encode($profile));
}else{
$this->send($fd, '550 Get profile fail!');
}
}else{
$this->send($fd, '500 Syntax error: GROUP-GET-PROFILE||{"group":""}');
}
break;
//Format: group-delete||{"group":""}
case 'GROUP-DELETE':
if(isset($data[1])){
$json = json_decode(trim($data[1]),true);
$group = isset($json['group'])?$json['group']:'';
if($this->user->delGroup($group)){
$this->user->save();
$this->user->reload();
$this->send($fd, '200 Group '.$group.' deleted.');
}else{
$this->send($fd, '550 Delete group fail!');
}
}else{
$this->send($fd, '500 Syntax error: GROUP-DELETE||{"group":""}');
}
break;
case 'GROUP-LIST':
$this->user->reload();
$list = $this->user->getGroupList();
$this->send($fd, '200 '.json_encode($list));
break;
//獲取組用戶列表
//Format: group-user-list||{"group":""}
case 'GROUP-USER-LIST':
if(isset($data[1])){
$json = json_decode(trim($data[1]),true);
$group = isset($json['group'])?$json['group']:'';
$this->user->reload();
$this->send($fd, '200 '.json_encode($this->user->getUserListOfGroup($group)));
}else{
$this->send($fd, '500 Syntax error: GROUP-USER-LIST||{"group":""}');
}
break;
// 獲取磁盤空間
//Format: disk-total||{"path":""}
case 'DISK-TOTAL':
if(isset($data[1])){
$json = json_decode(trim($data[1]),true);
$path = isset($json['path'])?$json['path']:'';
$size = 0;
if($path){
$size = disk_total_space($path);
}
$this->send($fd, '200 '.$size);
}else{
$this->send($fd, '500 Syntax error: DISK-TOTAL||{"path":""}');
}
break;
// 獲取磁盤空間
//Format: disk-total||{"path":""}
case 'DISK-FREE':
if(isset($data[1])){
$json = json_decode(trim($data[1]),true);
$path = isset($json['path'])?$json['path']:'';
$size = 0;
if($path){
$size = disk_free_space($path);
}
$this->send($fd, '200 '.$size);
}else{
$this->send($fd, '500 Syntax error: DISK-FREE||{"path":""}');
}
break;
case 'HELP':
$list = 'USER-ONLINE USER-ADD USER-SET-PROFILE USER-GET-PROFILE USER-DELETE USER-LIST GROUP-ADD GROUP-SET-PROFILE GROUP-GET-PROFILE GROUP-DELETE GROUP-LIST GROUP-USER-LIST DISK-TOTAL DISK-FREE';
$this->send($fd, '200 '.$list);
break;
default:
$this->send($fd, '500 Syntax error.');
}
} 
} 

總結:

至此,我們就可以實現(xiàn)一個完整的ftp服務器了。這個服務器的功能可以進行完全個性化定制。如果您有好的建議,也可以留言給我,謝謝。

相關文章

最新評論

一区二区三区另类在线| 成人18禁网站在线播放| 日韩中文字幕在线播放第二页| 成年人啪啪视频在线观看| 欧美中国日韩久久精品| 最新国产精品网址在线观看| 天天色天天舔天天射天天爽| 成人免费毛片aaaa| 日韩a级黄色小视频| 国产成人精品一区在线观看| 欧美3p在线观看一区二区三区| 中文字幕高清在线免费播放| 久久尻中国美女视频| 中文字幕日韩人妻在线三区| 三级av中文字幕在线观看| 亚洲成人激情av在线| 懂色av蜜桃a v| 欧美一区二区三区激情啪啪啪| 91社福利《在线观看| 插小穴高清无码中文字幕| 亚洲国产欧美国产综合在线| 国产之丝袜脚在线一区二区三区 | 成人高潮aa毛片免费| 东游记中文字幕版哪里可以看到| 男大肉棒猛烈插女免费视频| 在线免费91激情四射| 日韩精品中文字幕播放| 欧美成一区二区三区四区| okirakuhuhu在线观看| 亚洲在线一区二区欧美| 国产av欧美精品高潮网站| 日日日日日日日日夜夜夜夜夜夜| av在线观看网址av| 特一级特级黄色网片| 国产亚洲欧美45p| 天堂女人av一区二区| 又粗又硬又猛又爽又黄的| 久久久久国产成人精品亚洲午夜| 日美女屁股黄邑视频| 天天日天天日天天擦| 大白屁股精品视频国产| 一区二区视频在线观看免费观看| 天码人妻一区二区三区在线看| 国产精品成久久久久三级蜜臀av | 日本www中文字幕| 熟女国产一区亚洲中文字幕| 欧美一区二区三区四区性视频| 在线观看国产免费麻豆| 91香蕉成人app下载| 蜜臀成人av在线播放| 伊人成人在线综合网| aiss午夜免费视频| 国产午夜亚洲精品不卡在线观看| 色婷婷综合激情五月免费观看| 亚洲成人国产av在线| 亚洲人妻视频在线网| www,久久久,com| 色97视频在线播放| 日本韩国在线观看一区二区| 99的爱精品免费视频| 麻豆精品成人免费视频| 在线播放 日韩 av| 一区二区三区久久久91| 午夜影院在线观看视频羞羞羞| 人人妻人人澡人人爽人人dvl| 亚洲视频在线视频看视频在线| 丰满熟女午夜福利视频| 人妻少妇精品久久久久久| 91国语爽死我了不卡| 亚洲熟女综合色一区二区三区四区| 欧洲亚洲欧美日韩综合| av亚洲中文天堂字幕网| 午夜免费体验区在线观看| 快插进小逼里大鸡吧视频| 强行扒开双腿猛烈进入免费版| 亚洲一级av大片免费观看| 亚洲熟女综合色一区二区三区四区| 国产美女午夜福利久久| 国产一区av澳门在线观看| 亚洲国产成人最新资源| 亚洲欧美综合在线探花| 久久久久久久精品成人热| 日本熟妇色熟妇在线观看| 欧美80老妇人性视频| 在线视频自拍第三页| 亚洲最大黄 嗯色 操 啊| 日本欧美视频在线观看三区| 国产高清97在线观看视频| 丝袜国产专区在线观看| 国产九色91在线观看精品| 国产超码片内射在线| 国产中文字幕四区在线观看| 2019av在线视频| 亚洲蜜臀av一区二区三区九色| 国产日韩欧美视频在线导航| 在线 中文字幕 一区| 男人天堂最新地址av| 馒头大胆亚洲一区二区| 揄拍成人国产精品免费看视频| 亚洲自拍偷拍综合色| 亚洲精品国品乱码久久久久| 91免费放福利在线观看| 亚洲高清国产拍青青草原| 狠狠的往里顶撞h百合| 护士小嫩嫩又紧又爽20p| 鸡巴操逼一级黄色气| 97少妇精品在线观看| 精品一区二区三区午夜| 中文字幕一区二区自拍| 中字幕人妻熟女人妻a62v网| av破解版在线观看| 2017亚洲男人天堂| 亚洲欧美自拍另类图片| 日本欧美视频在线观看三区| 农村胖女人操逼视频| 欧美特色aaa大片| 国产精品免费不卡av| 日本www中文字幕| 2021国产一区二区| 久久免看30视频口爆视频| 国产伊人免费在线播放| 100%美女蜜桃视频| 扒开让我视频在线观看| 人妻熟女在线一区二区| 日本午夜福利免费视频| 亚洲日本一区二区三区| 在线观看黄色成年人网站| 日本黄色三级高清视频| 精品少妇一二三视频在线| 亚洲一区二区三区精品视频在线| rct470中文字幕在线| 欧美精品伦理三区四区| 少妇ww搡性bbb91| 天天干天天操天天爽天天摸| 午夜精品一区二区三区4| 在线观看一区二区三级| 欧美特色aaa大片| 美女张开腿让男生操在线看| 蜜臀av久久久久蜜臀av麻豆| 99精品视频在线观看免费播放| 欧美交性又色又爽又黄麻豆| 日韩一区二区电国产精品| 岛国免费大片在线观看| 自拍偷拍日韩欧美亚洲| 亚洲特黄aaaa片| 中文字幕第三十八页久久| 九色porny九色9l自拍视频| 搡老熟女一区二区在线观看| 91人妻精品久久久久久久网站| 国产之丝袜脚在线一区二区三区| 人人妻人人爱人人草| 女同互舔一区二区三区| 福利视频广场一区二区| 自拍 日韩 欧美激情| 新婚人妻聚会被中出| 免费黄页网站4188| 91国产在线免费播放| 免费无毒热热热热热热久| 第一福利视频在线观看| 国产亚洲精品视频合集| av天堂中文字幕最新| 免费看国产又粗又猛又爽又黄视频| 超级碰碰在线视频免费观看| www日韩a级s片av| 日本少妇的秘密免费视频| 人妻凌辱欧美丰满熟妇| 天天日天天添天天爽| av网站色偷偷婷婷网男人的天堂| 亚洲国产欧美一区二区丝袜黑人| 精品国产在线手机在线| 77久久久久国产精产品| 国产视频精品资源网站| 狠狠地躁夜夜躁日日躁| 在线国产日韩欧美视频| 婷婷综合蜜桃av在线| 亚洲麻豆一区二区三区| 色综合久久久久久久久中文| 精品国产在线手机在线| 亚洲成人激情av在线| 亚洲 清纯 国产com| 好男人视频在线免费观看网站| 国产精品黄片免费在线观看| 大鸡巴后入爆操大屁股美女| 老司机福利精品免费视频一区二区 | 国产女孩喷水在线观看| 色婷婷精品大在线观看| 19一区二区三区在线播放| 国产又色又刺激在线视频 | 美女张开两腿让男人桶av| 四川五十路熟女av| 国产精品成久久久久三级蜜臀av| 91国产在线视频免费观看| 51精品视频免费在线观看| 粉嫩av懂色av蜜臀av| 欧美精品免费aaaaaa| av完全免费在线观看av| 中文字幕在线观看国产片| 国产在线一区二区三区麻酥酥| 青青青视频手机在线观看| 91九色porny国产蝌蚪视频| 午夜激情久久不卡一区二区| 欧美va不卡视频在线观看| 91she九色精品国产| 午夜dv内射一区区| 天天插天天色天天日| 午夜青青草原网在线观看| 成人精品在线观看视频| 亚洲中文字幕校园春色| 色偷偷伊人大杳蕉综合网| 国产一区av澳门在线观看| 毛片av在线免费看| 欧美偷拍自拍色图片| 国产露脸对白在线观看| 色伦色伦777国产精品| 欧美日本在线视频一区| 欧美精品免费aaaaaa| 老司机福利精品视频在线| 亚洲精品福利网站图片| 视频二区在线视频观看| 久久久制服丝袜中文字幕| 亚洲粉嫩av一区二区三区| 老鸭窝在线观看一区| 日本丰满熟妇大屁股久久| 午夜精品久久久久久99热| 91免费放福利在线观看| 一区二区三区蜜臀在线| 国产自拍黄片在线观看| 中国黄色av一级片| 日韩中文字幕福利av| 欧美第一页在线免费观看视频| 成人免费做爰高潮视频| 亚洲 自拍 色综合图| 亚洲国际青青操综合网站| 大黑人性xxxxbbbb| 人妻少妇中文有码精品| 少妇高潮一区二区三区| 欧洲精品第一页欧洲精品亚洲| 国产一区成人在线观看视频| 中文亚洲欧美日韩无线码| 97青青青手机在线视频| 国产av一区2区3区| 二区中出在线观看老师| 亚洲午夜电影在线观看| 亚洲精品ww久久久久久| 国产精品成人xxxx| aⅴ五十路av熟女中出| 国产又粗又硬又大视频| 天堂av在线最新版在线| 一区二区三区欧美日韩高清播放 | 欧美伊人久久大香线蕉综合| 大香蕉伊人中文字幕| 日本少妇高清视频xxxxx| 美女大bxxxx内射| 无码中文字幕波多野不卡| 我想看操逼黄色大片| 国产女人露脸高潮对白视频| 亚洲欧洲av天堂综合| 免费男阳茎伸入女阳道视频| 91精品一区二区三区站长推荐| 美女在线观看日本亚洲一区| 国产精品大陆在线2019不卡| 大骚逼91抽插出水视频| 久草电影免费在线观看| 欧美美女人体视频一区| 欧美视频一区免费在线| 蜜臀av久久久久久久| 中文字幕在线观看国产片| 亚洲av日韩高清hd| 日本美女成人在线视频| 国产午夜福利av导航| 亚洲在线免费h观看网站| 国产精彩对白一区二区三区| 五十路老熟女码av| 家庭女教师中文字幕在线播放| 老司机午夜精品视频资源| 亚洲av色图18p| 国语对白xxxx乱大交| 亚洲成人国产综合一区| 日本午夜爽爽爽爽爽视频在线观看| 偷拍自拍国产在线视频| 狠狠躁狠狠爱网站视频| 日日夜夜狠狠干视频| 888欧美视频在线| 97人妻人人澡爽人人精品| 国产一区二区三免费视频 | 欧美美女人体视频一区| 精品少妇一二三视频在线| 在线国产精品一区二区三区| 中文字幕人妻被公上司喝醉在线| 国产精品一二三不卡带免费视频| 中国黄色av一级片| 又粗又长 明星操逼小视频| 日本成人一区二区不卡免费在线| 精品亚洲在线免费观看| 国产精品国产三级国产午| 3D动漫精品啪啪一区二区下载| 欧美综合婷婷欧美综合| 国产亚洲视频在线观看| 男人天堂色男人av| 99久久99一区二区三区| 天天躁日日躁狠狠躁躁欧美av| 三级等保密码要求条款| 欧美亚洲免费视频观看| 一区二区三区日本伦理| 亚洲福利精品福利精品福利| 久久这里只有精彩视频免费| av一本二本在线观看| 黑人性生活视频免费看| 首之国产AV医生和护士小芳| 亚洲av天堂在线播放| 福利视频一区二区三区筱慧| 国产成人小视频在线观看无遮挡 | 深田咏美亚洲一区二区| 人人妻人人爱人人草| 亚洲av色图18p| 2021久久免费视频| 东京热男人的av天堂| 欧美日韩亚洲国产无线码| 一区二区三区综合视频| 班长撕开乳罩揉我胸好爽| 成人亚洲精品国产精品 | yy6080国产在线视频| 亚洲日本一区二区久久久精品| 3337p日本欧洲大胆色噜噜| 天天插天天狠天天操| 亚洲av极品精品在线观看| 日本少妇人妻xxxxxhd| 青青青激情在线观看视频| 沈阳熟妇28厘米大战黑人| 熟女人妻一区二区精品视频| 一区二区三区毛片国产一区| 国产精品福利小视频a| 亚洲国产欧美一区二区丝袜黑人| 蜜臀成人av在线播放| 制服丝袜在线人妻中文字幕| 亚洲av在线观看尤物| 成人亚洲国产综合精品| 国产激情av网站在线观看| 天天干狠狠干天天操| 熟女俱乐部一二三区| 成人亚洲精品国产精品| 最近中文2019年在线看| 性感美女高潮视频久久久| 日本欧美视频在线观看三区| 欧美日韩不卡一区不区二区| 人人爽亚洲av人人爽av| 青青在线视频性感少妇和隔壁黑丝 | 91超碰青青中文字幕| 免费一级黄色av网站| 午夜精品一区二区三区福利视频| 中文字幕第1页av一天堂网| 亚洲国产精品免费在线观看| 91老熟女连续高潮对白| 女同互舔一区二区三区| 2017亚洲男人天堂| 天天干天天日天天谢综合156| 国产又粗又猛又爽又黄的视频在线| 久久热这里这里只有精品| 国产精品久久久久国产三级试频 | 国产女人露脸高潮对白视频| 国产乱子伦一二三区| 国产午夜激情福利小视频在线| 久久美欧人妻少妇一区二区三区| 国产午夜无码福利在线看| 亚洲熟妇x久久av久久| 福利视频网久久91| 国产亚洲国产av网站在线| 日韩中文字幕福利av| 巨乳人妻日下部加奈被邻居中出| 特大黑人巨大xxxx| 日韩欧美高清免费在线| 日本少妇的秘密免费视频| 老熟妇xxxhd老熟女| 国产又粗又猛又爽又黄的视频在线| 一区二区久久成人网| 在线视频这里只有精品自拍| 午夜成午夜成年片在线观看| 2018最新中文字幕在线观看| 中文字幕免费福利视频6| 制丝袜业一区二区三区| 国产又粗又猛又爽又黄的视频美国| jiuse91九色视频| 黄色片一级美女黄色片| 春色激情网欧美成人| 无码精品一区二区三区人| 偷拍自拍福利视频在线观看| 久久免看30视频口爆视频| 久久麻豆亚洲精品av| 91精品视频在线观看免费| 97人妻人人澡爽人人精品| 丰满的子国产在线观看| 日本av熟女在线视频| 亚洲人妻30pwc| 99热色原网这里只有精品| 国产精品久久久久国产三级试频| 成人久久精品一区二区三区| 欧美va不卡视频在线观看| 免费在线观看污污视频网站| 久久麻豆亚洲精品av| 天天干天天插天天谢| 日本xx片在线观看| 欧美在线精品一区二区三区视频| 久草极品美女视频在线观看| 天天日夜夜操天天摸| 久久久精品欧洲亚洲av| 人妻凌辱欧美丰满熟妇| 日本在线一区二区不卡视频| 中文字幕,亚洲人妻| 国产视频一区二区午夜| av久久精品北条麻妃av观看| 天天日天天摸天天爱| 亚洲欧美色一区二区| 免费黄色成人午夜在线网站| 成人动漫大肉棒插进去视频| 黄色大片男人操女人逼| av无限看熟女人妻另类av| 精产国品久久一二三产区区别| 色综合久久久久久久久中文| 精品suv一区二区69| 99的爱精品免费视频| 爆乳骚货内射骚货内射在线| 小泽玛利亚视频在线观看| rct470中文字幕在线| 天天操天天干天天艹| 亚洲码av无色中文| 激情人妻校园春色亚洲欧美| 视频久久久久久久人妻| 老司机午夜精品视频资源| 视频一区二区三区高清在线| 在线国产中文字幕视频| 动色av一区二区三区| 久久久噜噜噜久久熟女av| 天天日天天干天天搡| 玖玖一区二区在线观看| 日本xx片在线观看| 91在线视频在线精品3| 亚洲女人的天堂av| 亚洲成av人无码不卡影片一| 久久这里有免费精品| 99久久成人日韩欧美精品| 亚洲国产精品久久久久久6| 国产乱弄免费视频观看| 日韩成人免费电影二区| 国产精品国产三级国产午| 日本高清撒尿pissing| 天天操天天干天天艹| 天天干天天操天天玩天天射| 鸡巴操逼一级黄色气| 亚洲精品国产久久久久久| 亚洲麻豆一区二区三区| 成人国产影院在线观看| 青青草人人妻人人妻| 首之国产AV医生和护士小芳| 91麻豆精品传媒国产黄色片| 经典av尤物一区二区| 又黄又刺激的午夜小视频| 欧美特级特黄a大片免费| 国产一区自拍黄视频免费观看| okirakuhuhu在线观看| 日日操综合成人av| 无忧传媒在线观看视频| 日视频免费在线观看| 一区二区三区久久久91| 国产黄色片蝌蚪九色91| 欧洲日韩亚洲一区二区三区| 91色老99久久九九爱精品| 国产一线二线三线的区别在哪| yellow在线播放av啊啊啊| 亚洲熟妇x久久av久久| 国产精品欧美日韩区二区| 中文字幕AV在线免费看 | 久久久久久久久久性潮| 2022天天干天天操| 精品国产午夜视频一区二区| 天天日天天爽天天干| 国产麻豆剧传媒精品国产av蜜桃| 97人妻人人澡爽人人精品| 国产成人自拍视频在线免费观看| 亚洲偷自拍高清视频| 日本免费视频午夜福利视频| 国产综合视频在线看片| 国产伊人免费在线播放| 人妻在线精品录音叫床| 国产真实灌醉下药美女av福利| 欧洲精品第一页欧洲精品亚洲 | 欧美专区日韩专区国产专区| 国产精品国产三级国产精东| 国产在线91观看免费观看| 欧美日本aⅴ免费视频| 夜夜嗨av蜜臀av| 精品亚洲在线免费观看| 一区二区三区 自拍偷拍| 一区二区三区另类在线| 中文字幕欧美日韩射射一| 亚洲最大免费在线观看| 天美传媒mv视频在线观看| 亚洲另类综合一区小说| 91试看福利一分钟| aiss午夜免费视频| 9色在线视频免费观看| 动漫精品视频在线观看| 一区二区三区在线视频福利| 99热这里只有精品中文| 亚洲精品福利网站图片| 亚洲伊人av天堂有码在线| 中文字幕人妻一区二区视频| chinese国产盗摄一区二区| 人妻少妇精品久久久久久| 亚洲 中文字幕在线 日韩| 国产精品久久久黄网站| 亚洲天堂有码中文字幕视频| 欧美精品中文字幕久久二区| 五月色婷婷综合开心网4438| 亚洲精品在线资源站| 中文字幕之无码色多多| 国产精品一二三不卡带免费视频| 中文字幕成人日韩欧美| 91色网站免费在线观看| 亚洲国产40页第21页| 大黑人性xxxxbbbb| 可以在线观看的av中文字幕| 激情色图一区二区三区| 97超碰免费在线视频| 888欧美视频在线| 一二三中文乱码亚洲乱码one| 2020国产在线不卡视频 | 2o22av在线视频| 老师啊太大了啊啊啊尻视频| 亚洲视频在线观看高清| 91精品国产观看免费| 精品国产在线手机在线| 传媒在线播放国产精品一区| 偷拍3456eee| 亚洲人人妻一区二区三区| 国产白袜脚足J棉袜在线观看| 亚洲av日韩av网站| 国产麻豆精品人妻av| www日韩a级s片av| 2022精品久久久久久中文字幕| 欧美精产国品一二三区| 天天日天天鲁天天操| 老鸭窝日韩精品视频观看| 国产午夜激情福利小视频在线| 国产精品国产三级国产午| 中文字幕 码 在线视频| 精品一区二区三区午夜| 特大黑人巨大xxxx| 欧美男人大鸡吧插女人视频| av视屏免费在线播放| 色婷婷综合激情五月免费观看| 成年午夜免费无码区| 九九热99视频在线观看97| 在线制服丝袜中文字幕| caoporm超碰国产| 亚洲熟妇x久久av久久| okirakuhuhu在线观看| av破解版在线观看| 美女福利写真在线观看视频| 国产高清97在线观看视频| 无码国产精品一区二区高潮久久4| 青青青艹视频在线观看| 午夜美女福利小视频| 91九色porny国产蝌蚪视频| 国产使劲操在线播放| 日韩三级黄色片网站| 五月精品丁香久久久久福利社| 99婷婷在线观看视频| 啪啪啪18禁一区二区三区| 99国内小视频在现欢看| 欧美地区一二三专区| 视频久久久久久久人妻| 国产视频网站国产视频| 青青青青操在线观看免费| 老司机福利精品免费视频一区二区| 色狠狠av线不卡香蕉一区二区| 精品一区二区三区三区色爱| 都市激情校园春色狠狠| 亚洲欧美综合在线探花| 亚洲国产欧美国产综合在线| 青娱乐最新视频在线| av无限看熟女人妻另类av| 亚洲成人国产综合一区| 一区二区三区日韩久久| 337p日本大胆欧美人| 天码人妻一区二区三区在线看| 成人免费毛片aaaa| 美女小视频网站在线| 一二三中文乱码亚洲乱码one| 国产精品国产精品一区二区| 男人操女人的逼免费视频| 亚洲国际青青操综合网站| 91‖亚洲‖国产熟女| yellow在线播放av啊啊啊| 亚洲人妻国产精品综合| 视频 一区二区在线观看| 日本高清成人一区二区三区| 日韩精品一区二区三区在线播放| 国产亚州色婷婷久久99精品| 偷拍自拍国产在线视频| 97国产在线观看高清| 啪啪啪啪啪啪啪啪啪啪黄色| av在线观看网址av| 超碰97人人做人人爱| 宅男噜噜噜666国产| av在线免费资源站| 日本女人一级免费片| 青青青青青操视频在线观看| 亚洲午夜福利中文乱码字幕| www天堂在线久久| 成人国产小视频在线观看| 又粗又长 明星操逼小视频| 中文字幕中文字幕 亚洲国产| 伊人成人在线综合网| 日本精品视频不卡一二三| 一区二区免费高清黄色视频| 国产之丝袜脚在线一区二区三区| 日韩美女综合中文字幕pp| 五月激情婷婷久久综合网| 亚洲成人精品女人久久久| 午夜频道成人在线91| 在线观看的黄色免费网站| 五十路av熟女松本翔子| 摧残蹂躏av一二三区| 精品av久久久久久久| av一本二本在线观看| 啊慢点鸡巴太大了啊舒服视频| 久精品人妻一区二区三区| 免费一级特黄特色大片在线观看| 好吊操视频这里只有精品| 人人妻人人爽人人澡人人精品| 视频啪啪啪免费观看| 久草视频在线免播放| 97青青青手机在线视频 | 日本xx片在线观看| 男生舔女生逼逼视频| www骚国产精品视频| 日韩视频一区二区免费观看| 日本乱人一区二区三区| 91高清成人在线视频| 日本高清成人一区二区三区| 青青草人人妻人人妻| 姐姐的朋友2在线观看中文字幕| 美女福利视频导航网站| 久久久久国产成人精品亚洲午夜| 亚洲精品色在线观看视频| 欧美黄片精彩在线免费观看| 亚洲精品成人网久久久久久小说| 中国熟女一区二区性xx| 大鸡巴后入爆操大屁股美女| 老司机你懂得福利视频| 爱有来生高清在线中文字幕| 中文字幕一区二区三区人妻大片| 免费黄页网站4188| 黄色男人的天堂视频| 91久久国产成人免费网站| 哥哥姐姐综合激情小说| 91精品综合久久久久3d动漫| 亚洲特黄aaaa片| 亚洲欧美国产综合777| 欧洲日韩亚洲一区二区三区| 国产精品视频男人的天堂| 中文字幕无码日韩专区免费| 国产高清精品一区二区三区| 97超碰免费在线视频| 国产精品久久久黄网站| 护士小嫩嫩又紧又爽20p| 91chinese在线视频| 婷婷五月亚洲综合在线| 久久久久久国产精品| 喷水视频在线观看这里只有精品| 亚洲第一黄色在线观看| 国产精品中文av在线播放| 91香蕉成人app下载| 一区二区三区麻豆福利视频| 午夜婷婷在线观看视频| 红杏久久av人妻一区| 亚洲av成人免费网站| 日本av高清免费网站| 婷婷六月天中文字幕| 日本啪啪啪啪啪啪啪| 自拍偷拍亚洲另类色图| 国产欧美日韩第三页| 亚洲熟妇久久无码精品| 国产中文精品在线观看| 欧美日韩精品永久免费网址| 一区二区三区综合视频| 亚洲老熟妇日本老妇| 中文字幕日韩91人妻在线| 女同性ⅹxx女同h偷拍| 国产精品福利小视频a| 日本熟女50视频免费| 999久久久久999| 色综合天天综合网国产成人| 狠狠嗨日韩综合久久| 91免费观看国产免费| 日韩欧美一级精品在线观看| 国产午夜男女爽爽爽爽爽视频| 国产妇女自拍区在线观看| 91p0rny九色露脸熟女| heyzo蜜桃熟女人妻| 亚洲色偷偷综合亚洲AV伊人| 91桃色成人网络在线观看| 人妻丰满熟妇综合网| av视屏免费在线播放| 亚洲成人情色电影在线观看| 2020中文字幕在线播放| 日韩美女精品视频在线观看网站| 偷拍美女一区二区三区| 孕妇奶水仑乱A级毛片免费看| 视频二区在线视频观看| 91久久国产成人免费网站| 2012中文字幕在线高清| 青娱乐蜜桃臀av色| 伊人开心婷婷国产av| 91人妻精品一区二区在线看| 人妻丝袜av在线播放网址| 亚洲天堂精品久久久| 一级黄片大鸡巴插入美女| 日本一区二区三区免费小视频| 欧美成人综合色在线噜噜| 欧美日韩v中文在线| av视网站在线观看| 五月天久久激情视频| 日韩欧美中文国产在线| 男人天堂色男人av| 国产普通话插插视频| 色综合天天综合网国产成人| 日本少妇高清视频xxxxx| 国产精品黄片免费在线观看| 91福利在线视频免费观看| 亚洲第一伊人天堂网| 夜夜骑夜夜操夜夜奸| 偷拍自拍国产在线视频| 欧美日本国产自视大全| 国产精品成久久久久三级蜜臀av | 精产国品久久一二三产区区别 | 国产精品欧美日韩区二区| 亚洲精品乱码久久久久久密桃明| 中文字幕—97超碰网| 888亚洲欧美国产va在线播放| 亚洲成人国产av在线| 熟女国产一区亚洲中文字幕| 中文字幕1卡1区2区3区| 狠狠操操操操操操操操操| 黑人变态深video特大巨大| jul—619中文字幕在线| 国产日韩欧美视频在线导航| 亚洲天天干 夜夜操| 国产+亚洲+欧美+另类| AV天堂一区二区免费试看| 成人18禁网站在线播放| 亚洲图库另类图片区| 又粗又硬又猛又爽又黄的| 欧美女同性恋免费a| 操人妻嗷嗷叫视频一区二区| 大鸡巴插入美女黑黑的阴毛| 国产精选一区在线播放| 国产真实乱子伦a视频| 2021天天色天天干| 日本后入视频在线观看| 国产麻豆精品人妻av| 久草极品美女视频在线观看| 骚逼被大屌狂草视频免费看| 一区二区三区蜜臀在线| 成人影片高清在线观看 | 动漫黑丝美女的鸡巴| 沙月文乃人妻侵犯中文字幕在线| 天天草天天色天天干| 91麻豆精品91久久久久同性| 日韩成人性色生活片| 亚洲一区二区三区精品乱码| 亚洲午夜福利中文乱码字幕| 中文字幕高清资源站| 精产国品久久一二三产区区别| 亚洲福利天堂久久久久久| 老司机午夜精品视频资源| 大鸡巴插入美女黑黑的阴毛| 久久久久久久99精品| 欧美亚洲自偷自拍 在线| 天天干狠狠干天天操| av成人在线观看一区| av完全免费在线观看av| 青青草国内在线视频精选| 亚洲 国产 成人 在线| 成人高清在线观看视频| 日韩a级精品一区二区| 亚洲欧洲一区二区在线观看| 成人免费做爰高潮视频| 日韩加勒比东京热二区| 免费十精品十国产网站| 在线观看成人国产电影| 黑人大几巴狂插日本少妇| 亚洲变态另类色图天堂网| 亚洲成人av一区久久| 91桃色成人网络在线观看| 日本在线不卡免费视频| 日韩精品电影亚洲一区| 白白操白白色在线免费视频| 免费无码人妻日韩精品一区二区| 国产自拍在线观看成人| 欧美在线一二三视频| 国产之丝袜脚在线一区二区三区 | 成人国产小视频在线观看| 亚洲天堂精品福利成人av| www,久久久,com| jiujiure精品视频在线| 可以在线观看的av中文字幕| 日韩精品激情在线观看| 亚洲国产香蕉视频在线播放| av老司机亚洲一区二区| 亚洲综合自拍视频一区| 日本高清成人一区二区三区| 99热久久这里只有精品8| 亚洲av日韩精品久久久久久hd| 午夜精品在线视频一区| 天天日天天干天天舔天天射| 成人24小时免费视频| 91精品国产观看免费| 欧美精品 日韩国产| 被大鸡吧操的好舒服视频免费| 热久久只有这里有精品| 黄色大片免费观看网站| av线天堂在线观看| 亚洲成av人无码不卡影片一| 美女在线观看日本亚洲一区| 日本人妻少妇18—xx| 一级a看免费观看网站| 日韩视频一区二区免费观看| 亚洲日本一区二区三区| 亚洲精品乱码久久久久久密桃明| 韩国男女黄色在线观看| 亚洲综合自拍视频一区| 国语对白xxxx乱大交| 成人网18免费视频版国产| 亚洲成人激情av在线| 2020久久躁狠狠躁夜夜躁 | 中文字幕一区二区三区人妻大片| 欧美麻豆av在线播放| 亚洲青青操骚货在线视频| 成人国产影院在线观看| 五月婷婷在线观看视频免费| 最新日韩av传媒在线| 亚洲人成精品久久久久久久| 日本精品一区二区三区在线视频。| 亚洲天堂有码中文字幕视频| 91大屁股国产一区二区| chinese国产盗摄一区二区| 欧美老鸡巴日小嫩逼| 啪啪啪18禁一区二区三区| 国产精品一二三不卡带免费视频 | 亚洲午夜精品小视频| 国产精品一区二区av国| 日本一道二三区视频久久| 亚洲国产精品免费在线观看| 亚洲午夜高清在线观看| 亚洲精品 欧美日韩| 国产精品黄片免费在线观看| 在线不卡成人黄色精品| jiuse91九色视频| 亚洲综合在线观看免费| 日韩在线中文字幕色| 在线免费观看视频一二区| 狠狠躁夜夜躁人人爽天天天天97| 日本一区精品视频在线观看| 青青草亚洲国产精品视频| 97黄网站在线观看| 扒开让我视频在线观看| 又粗又长 明星操逼小视频| 91麻豆精品传媒国产黄色片| 青娱乐最新视频在线| av网站色偷偷婷婷网男人的天堂| 天天干天天插天天谢| 亚洲国产第一页在线观看| 久久美欧人妻少妇一区二区三区| 99热久久极品热亚洲| 国产精品国产精品一区二区| 中文字幕日韩精品日本| 婷婷久久一区二区字幕网址你懂得 | 馒头大胆亚洲一区二区| 97资源人妻免费在线视频| 成人激情文学网人妻| 精品一区二区三区三区色爱| 欧洲精品第一页欧洲精品亚洲| 夫妻在线观看视频91| 国产一区二区视频观看| 99久久99一区二区三区| 蜜桃色婷婷久久久福利在线| 人人妻人人爽人人澡人人精品| 久久久久久久精品成人热| 18禁美女羞羞免费网站| 欧美日韩高清午夜蜜桃大香蕉| 一区二区三区久久中文字幕| 视频久久久久久久人妻| 欧美一区二区三区乱码在线播放 | 伊人精品福利综合导航| 粉嫩av蜜乳av蜜臀| 97超碰国语国产97超碰| 国产a级毛久久久久精品| 日本xx片在线观看| 日韩写真福利视频在线观看| 99精品免费观看视频| 国产一级精品综合av| 亚洲精品国产在线电影| 日韩成人综艺在线播放| 最近中文2019年在线看| 北条麻妃高跟丝袜啪啪| 97年大学生大白天操逼| 超碰97人人澡人人| 一区二区在线观看少妇| 国产女孩喷水在线观看| 亚洲精品一线二线在线观看| 久草视频在线免播放| 中文字幕日本人妻中出| 家庭女教师中文字幕在线播放| 精品黑人一区二区三区久久国产| 91 亚洲视频在线观看| 成人资源在线观看免费官网| 女生被男生插的视频网站| 大陆精品一区二区三区久久| 99热久久这里只有精品| 欧美黑人性暴力猛交喷水| 又大又湿又爽又紧A视频| 粉嫩av蜜乳av蜜臀| 男生舔女生逼逼视频| 2019av在线视频| 亚洲欧美国产综合777| 青青在线视频性感少妇和隔壁黑丝| av视屏免费在线播放| 沙月文乃人妻侵犯中文字幕在线 | 日本一区二区三区免费小视频| 国产精品成人xxxx| 黄色中文字幕在线播放| 99亚洲美女一区二区三区| 色秀欧美视频第一页| 欧美特级特黄a大片免费| 91色网站免费在线观看| 在线国产日韩欧美视频| 又大又湿又爽又紧A视频| 日韩美女综合中文字幕pp| ka0ri在线视频| 久草视频在线免播放| 国产妇女自拍区在线观看| 亚洲熟女久久久36d| 亚洲va天堂va国产va久| 孕妇奶水仑乱A级毛片免费看| 在线成人日韩av电影| 大胆亚洲av日韩av| 40道精品招牌菜特色| 国产精品国产三级麻豆| 国产精品精品精品999| 精品高潮呻吟久久av| 欧美黑人性暴力猛交喷水| 日美女屁股黄邑视频| 国产精品精品精品999| 一区二区视频在线观看免费观看| 天堂av中文在线最新版| 中文字幕 人妻精品| 亚洲午夜电影之麻豆| 天天草天天色天天干| 亚洲va天堂va国产va久| 欲乱人妻少妇在线视频裸| 人人妻人人爽人人澡人人精品| 社区自拍揄拍尻屁你懂的| 揄拍成人国产精品免费看视频| 日韩中文字幕在线播放第二页| 狠狠地躁夜夜躁日日躁| 啊啊好慢点插舔我逼啊啊啊视频| 天天躁日日躁狠狠躁av麻豆| 久草视频福利在线首页| 老司机99精品视频在线观看| 亚洲福利精品福利精品福利| 都市激情校园春色狠狠| 亚洲精品高清自拍av| 欧美日韩一区二区电影在线观看| 日本高清撒尿pissing| 亚洲高清视频在线不卡| 日韩av有码中文字幕| 国产一区二区在线欧美| 成人高潮aa毛片免费| caoporn蜜桃视频| 五十路丰满人妻熟妇| 无码精品一区二区三区人| 国产熟妇乱妇熟色T区| 精品区一区二区三区四区人妻| 狠狠操狠狠操免费视频| 国产亚洲欧美另类在线观看| 欧美中文字幕一区最新网址| 亚洲区欧美区另类最新章节| 成人av中文字幕一区| 免费成人av中文字幕| 亚洲天堂有码中文字幕视频| 欧美一区二区三区高清不卡tv| japanese五十路熟女熟妇| 久久久久久久精品老熟妇| 亚洲一级特黄特黄黄色录像片| 成人亚洲国产综合精品| 国产精品久久综合久久| 亚洲免费福利一区二区三区| 首之国产AV医生和护士小芳| 中字幕人妻熟女人妻a62v网| 亚洲高清国产拍青青草原| 日韩加勒比东京热二区| 91www一区二区三区| 国产一区av澳门在线观看| 在线国产精品一区二区三区| 欧美少妇性一区二区三区| 99精品一区二区三区的区| 亚洲高清自偷揄拍自拍| 国产91久久精品一区二区字幕| 中英文字幕av一区| 亚洲欧美成人综合在线观看| 欧亚日韩一区二区三区观看视频 | 黑人巨大精品欧美视频| 欧美日韩国产一区二区三区三州 | 操人妻嗷嗷叫视频一区二区 | 日韩美在线观看视频黄| 激情图片日韩欧美人妻| 成人影片高清在线观看| 国产精品日韩欧美一区二区| 国产高清在线在线视频| 伊人情人综合成人久久网小说 | 国产成人精品av网站| 韩国AV无码不卡在线播放| 亚洲av自拍天堂网| 亚洲一级av大片免费观看| 岛国毛片视频免费在线观看| 欧美日韩亚洲国产无线码| 无码国产精品一区二区高潮久久4 日韩欧美一级精品在线观看 | 和邻居少妇愉情中文字幕| 9色在线视频免费观看| 亚洲天堂成人在线观看视频网站| 9色精品视频在线观看| 国产熟妇一区二区三区av| www骚国产精品视频| 亚洲福利精品视频在线免费观看| 桃色视频在线观看一区二区| 天天艹天天干天天操| 大屁股熟女一区二区三区| 在线成人日韩av电影| 手机看片福利盒子日韩在线播放| 国产成人精品福利短视频| 天天日天天天天天天天天天天 | 男人插女人视频网站| 国产熟妇乱妇熟色T区| 天天日天天鲁天天操| 天天操天天弄天天射| 黄色在线观看免费观看在线| 精品一区二区三区三区88| 在线观看视频一区麻豆| 精品国产亚洲av一淫| 在线观看国产免费麻豆| 九色porny九色9l自拍视频| 91小伙伴中女熟女高潮| 丝袜长腿第一页在线| 日本性感美女视频网站| 班长撕开乳罩揉我胸好爽| 日本乱人一区二区三区| 大鸡巴操娇小玲珑的女孩逼| 在线视频国产欧美日韩| 青青尤物在线观看视频网站| 日韩欧美一级精品在线观看| 丰满的继坶3中文在线观看| 93精品视频在线观看| 久久久久久久久久性潮| 老司机午夜精品视频资源| 日韩av中文在线免费观看| 国产欧美精品免费观看视频| 狠狠鲁狠狠操天天晚上干干| 4个黑人操素人视频网站精品91| 色在线观看视频免费的| 欧美国产亚洲中英文字幕| 一二三中文乱码亚洲乱码one| 亚洲熟妇久久无码精品| 亚洲av日韩av网站| 久久久久只精品国产三级| 岛国av高清在线成人在线| 成熟丰满熟妇高潮xx×xx| 国产日韩欧美美利坚蜜臀懂色| 久久久久久久一区二区三| 狠狠躁夜夜躁人人爽天天天天97| 狠狠鲁狠狠操天天晚上干干| 欧美日本aⅴ免费视频| 涩爱综合久久五月蜜臀| 亚洲免费视频欧洲免费视频| 91av精品视频在线| 日本熟女精品一区二区三区| 97黄网站在线观看| 春色激情网欧美成人| 91色九色porny| 日韩欧美一级黄片亚洲| 国产视频在线视频播放| 人妻久久无码中文成人| 日韩三级黄色片网站| weyvv5国产成人精品的视频| 端庄人妻堕落挣扎沉沦| 欧美视频一区免费在线| av在线免费资源站| 青青青青青青草国产| 国产露脸对白在线观看| 黄色资源视频网站日韩| 久久热这里这里只有精品| 国产精品久久综合久久| 精内国产乱码久久久久久| 天天通天天透天天插| 中文字幕一区二区人妻电影冢本| 91成人在线观看免费视频| 66久久久久久久久久久| 初美沙希中文字幕在线| 成人av天堂丝袜在线观看| 中文字幕奴隷色的舞台50| 2022天天干天天操| 性欧美激情久久久久久久| h国产小视频福利在线观看| 2022天天干天天操| 中文字幕免费福利视频6| 亚洲一区二区三区久久午夜| 欧美视频一区免费在线| 人人妻人人人操人人人爽| 1769国产精品视频免费观看| 日韩欧美国产一区ab| 国产欧美日韩在线观看不卡| 亚洲国产成人在线一区| 91亚洲国产成人精品性色| 亚洲国产免费av一区二区三区| 不卡一区一区三区在线| 亚洲中文字幕综合小综合| 五十路熟女av天堂| 日本午夜福利免费视频| 哥哥姐姐综合激情小说| 桃色视频在线观看一区二区| 人妻少妇精品久久久久久| 日韩av有码中文字幕| 在线观看免费视频网| 亚洲午夜电影在线观看| 中文字幕在线观看国产片| av在线观看网址av| 国产刺激激情美女网站| 国产品国产三级国产普通话三级| 极品粉嫩小泬白浆20p主播| 成人伊人精品色xxxx视频| 91免费观看国产免费| 都市家庭人妻激情自拍视频| 国产精品国产三级麻豆| 国产亚洲四十路五十路| 91桃色成人网络在线观看| 97人妻夜夜爽二区欧美极品| 熟女少妇激情五十路| 免费国产性生活视频| 97色视频在线观看| 日韩人妻在线视频免费| 在线观看免费视频色97| 亚洲人妻30pwc| 人妻久久久精品69系列| 一区二区三区在线视频福利| 欧美日韩中文字幕欧美| av乱码一区二区三区| 视频二区在线视频观看| 天天日夜夜干天天操| 日日操综合成人av| 精品人妻伦一二三区久 | 精品欧美一区二区vr在线观看 | 国产亚洲欧美视频网站| 欧美精品资源在线观看| 在线免费观看日本片| 夜色17s精品人妻熟女| 欧美国品一二三产区区别 | avjpm亚洲伊人久久| 色婷婷六月亚洲综合香蕉| 不卡精品视频在线观看| 男生舔女生逼逼的视频| 日韩欧美一级aa大片| 1000部国产精品成人观看视频| 又黄又刺激的午夜小视频| 日本精品视频不卡一二三| 天天干天天操天天插天天日| 亚洲午夜精品小视频| 亚洲高清国产拍青青草原| 日韩熟女av天堂系列| av网站色偷偷婷婷网男人的天堂| 日韩三级电影华丽的外出| 久久丁香婷婷六月天| 91人妻精品一区二区久久| 亚洲无码一区在线影院| 2022国产精品视频| 97年大学生大白天操逼| 国产视频网站国产视频| 果冻传媒av一区二区三区| 欧美aa一级一区三区四区| 大鸡巴操b视频在线| 91av精品视频在线| 2022精品久久久久久中文字幕| 亚洲综合乱码一区二区| 亚洲欧美成人综合在线观看| nagger可以指黑人吗| www日韩a级s片av| 亚洲国产精品中文字幕网站| 美女日逼视频免费观看| av中文字幕网址在线| 丝袜美腿欧美另类 中文字幕| 亚洲av日韩精品久久久久久hd| 91福利在线视频免费观看| 黑人巨大的吊bdsm| 免费看高清av的网站| 99精品视频在线观看免费播放| 欧美区一区二区三视频| 少妇系列一区二区三区视频| 18禁污污污app下载| 欧美男同性恋69视频| 免费黄高清无码国产| 中文字幕免费在线免费| 2021年国产精品自拍| 在线免费观看日本伦理| 亚洲av香蕉一区区二区三区犇| 98精产国品一二三产区区别| 999久久久久999| 日韩精品中文字幕在线| 经典av尤物一区二区| av手机在线观播放网站| 欧美日韩激情啪啪啪 | 国产午夜无码福利在线看| 中文字幕免费在线免费| 91国内视频在线观看| 亚洲精品成人网久久久久久小说| 91成人在线观看免费视频| 青青青青青青青青青青草青青| 男人的网址你懂的亚洲欧洲av| 日本丰满熟妇大屁股久久| 亚洲熟女女同志女同| 99国内小视频在现欢看| 天天色天天舔天天射天天爽| 亚洲国产精品中文字幕网站| 亚洲第17页国产精品| 国产一线二线三线的区别在哪| 日韩成人综艺在线播放| 色综合天天综合网国产成人 | 欧美伊人久久大香线蕉综合| 小穴多水久久精品免费看| 欧美日韩中文字幕欧美| 日韩欧美一级精品在线观看| 91传媒一区二区三区| 最新国产亚洲精品中文在线| 日韩av有码一区二区三区4| 日韩欧美在线观看不卡一区二区 | 国产午夜男女爽爽爽爽爽视频| 欧美一区二区三区乱码在线播放| 日本啪啪啪啪啪啪啪| 白嫩白嫩美女极品国产在线观看| 青青在线视频性感少妇和隔壁黑丝 | 欧美xxx成人在线| 超级av免费观看一区二区三区| 懂色av蜜桃a v| 久久h视频在线观看| 阴茎插到阴道里面的视频| 亚洲激情av一区二区| 18禁美女羞羞免费网站| 绝色少妇高潮3在线观看| 午夜精品一区二区三区更新| 日本一道二三区视频久久| 国产精品成人xxxx| 精品日产卡一卡二卡国色天香| 亚洲图片欧美校园春色| 大白屁股精品视频国产| 蜜桃精品久久久一区二区| 国产高清97在线观看视频| 久青青草视频手机在线免费观看 | 免费观看丰满少妇做受| 桃色视频在线观看一区二区| 亚洲人妻国产精品综合| 午夜场射精嗯嗯啊啊视频| 成人av亚洲一区二区| 国产janese在线播放| 女警官打开双腿沦为性奴| 欧美视频中文一区二区三区| 国产视频在线视频播放| 少妇被强干到高潮视频在线观看| 国产揄拍高清国内精品对白| 福利视频广场一区二区| 少妇人妻100系列| 天天艹天天干天天操| 在线观看的a站 最新| 国产日韩欧美视频在线导航| 精品一区二区三区三区88 | 男生舔女生逼逼视频| 国产欧美精品一区二区高清| 欧美日本在线观看一区二区| 中文字幕 人妻精品| 操人妻嗷嗷叫视频一区二区| 综合页自拍视频在线播放| gay gay男男瑟瑟在线网站| 99热国产精品666| 夜色17s精品人妻熟女| 中国视频一区二区三区| 天堂av狠狠操蜜桃| 国产成人自拍视频在线免费观看| 欧美精品久久久久久影院| yy96视频在线观看| 亚洲免费av在线视频| 日本福利午夜电影在线观看| 日本免费一级黄色录像| 久久人人做人人妻人人玩精品vr| 欧美日韩情色在线观看| 免费福利av在线一区二区三区| 偷拍自拍 中文字幕| 91啪国自产中文字幕在线| 日本性感美女写真视频| av久久精品北条麻妃av观看| 激情五月婷婷免费视频| 日韩剧情片电影在线收看| 2021国产一区二区| 一级黄片久久久久久久久| 天天干夜夜操天天舔| 午夜国产免费福利av| 成熟熟女国产精品一区| 污污小视频91在线观看| 亚洲国产最大av综合| 蜜桃精品久久久一区二区| 国产麻豆乱子伦午夜视频观看| 成年人黄视频在线观看| 日本在线一区二区不卡视频| 成年人啪啪视频在线观看| 欧美精品欧美极品欧美视频 | 狠狠躁夜夜躁人人爽天天天天97| 美女被肏内射视频网站| 黄工厂精品视频在线观看| 制服丝袜在线人妻中文字幕| 女警官打开双腿沦为性奴| 亚洲成人黄色一区二区三区 | 色狠狠av线不卡香蕉一区二区| 国产日韩av一区二区在线| 青青青青爽手机在线| 天天插天天色天天日| 日本午夜爽爽爽爽爽视频在线观看| 女同性ⅹxx女同hd| 欧洲国产成人精品91铁牛tv| 中文字幕AV在线免费看 | 初美沙希中文字幕在线| 东游记中文字幕版哪里可以看到| 日韩美av高清在线| 动漫黑丝美女的鸡巴| 懂色av之国产精品| 99久久中文字幕一本人| 欧美精品黑人性xxxx| 国产一区av澳门在线观看| 国产黄色大片在线免费播放| av俺也去在线播放| 国产女人被做到高潮免费视频| 成人乱码一区二区三区av| 国产三级影院在线观看| 激情五月婷婷综合色啪| 涩涩的视频在线观看视频| 日本三极片视频网站观看| 91久久综合男人天堂| 一区二区三区视频,福利一区二区| 女生被男生插的视频网站| av视网站在线观看| 国产揄拍高清国内精品对白| 大陆胖女人与丈夫操b国语高清| 操日韩美女视频在线免费看| 国产成人精品一区在线观看| 亚洲成人情色电影在线观看| 亚洲av一妻不如妾| av天堂中文字幕最新| 一个色综合男人天堂| 天堂av狠狠操蜜桃| 国产精品日韩欧美一区二区| 美女福利视频网址导航| 在线不卡日韩视频播放| 国产乱弄免费视频观看| 在线免费观看国产精品黄色| 婷婷综合蜜桃av在线| 亚洲精品久久综合久| 91精品国产观看免费| 国产+亚洲+欧美+另类| 亚洲午夜福利中文乱码字幕| 伊人综合aⅴ在线网| 国产一区av澳门在线观看| 欧美亚洲少妇福利视频| 日本熟妇色熟妇在线观看| 亚洲欧美清纯唯美另类| 天天插天天色天天日| 在线播放一区二区三区Av无码| 成熟丰满熟妇高潮xx×xx| 99热这里只有精品中文| 久草电影免费在线观看| 亚洲专区激情在线观看视频| 91精品国产综合久久久蜜| 亚洲欧美激情中文字幕| 2017亚洲男人天堂| 91麻豆精品传媒国产黄色片| 天天摸天天亲天天舔天天操天天爽| 超碰97免费人妻麻豆| 亚洲成人av在线一区二区| 国产片免费观看在线观看| 天堂av在线播放免费| 91免费福利网91麻豆国产精品| 少妇人妻100系列| 91人妻精品一区二区在线看| 91色网站免费在线观看| 亚洲男人在线天堂网| 综合国产成人在线观看| 3344免费偷拍视频| 亚洲在线免费h观看网站| 男人操女人逼逼视频网站| 欧美精品中文字幕久久二区| 久久精品国产亚洲精品166m| 亚洲av日韩av第一区二区三区| 欧美专区日韩专区国产专区| 啊慢点鸡巴太大了啊舒服视频| 亚洲成人av在线一区二区| 欧洲黄页网免费观看| 人人人妻人人澡人人| 亚洲精品国偷自产在线观看蜜桃| 日本午夜爽爽爽爽爽视频在线观看| 午夜在线一区二区免费| 99热99re在线播放| 91人妻人人做人人爽在线| 欧洲欧美日韩国产在线| 免费黄色成人午夜在线网站| 天天操天天爽天天干| 亚洲成人av一区在线| 五月天色婷婷在线观看视频免费| 大鸡巴操b视频在线| 国产精品黄片免费在线观看| 亚洲精品一线二线在线观看| 日本高清在线不卡一区二区| 国产熟妇一区二区三区av| 极品丝袜一区二区三区| 亚洲午夜电影之麻豆| 直接能看的国产av| 爆乳骚货内射骚货内射在线| 欧美精产国品一二三产品价格| 亚洲精品国偷自产在线观看蜜桃| 男生舔女生逼逼视频| 女同久久精品秋霞网| japanese五十路熟女熟妇| 蜜桃视频入口久久久| 女警官打开双腿沦为性奴| 国产午夜无码福利在线看| 天天干天天操天天玩天天射| 成人性爱在线看四区| 99精品免费观看视频| 顶级尤物粉嫩小尤物网站| 午夜精品九一唐人麻豆嫩草成人 | 亚洲高清国产自产av| 啪啪啪18禁一区二区三区| 亚洲少妇人妻无码精品| 国产精品视频资源在线播放| 中文字幕奴隷色的舞台50| 在线 中文字幕 一区| 天天色天天操天天舔| 老熟妇xxxhd老熟女| 日韩精品电影亚洲一区| 欧美老鸡巴日小嫩逼| 丝袜国产专区在线观看| 2020久久躁狠狠躁夜夜躁| 在线视频国产欧美日韩| 瑟瑟视频在线观看免费视频| 啪啪啪啪啪啪啪啪啪啪黄色| 国产白袜脚足J棉袜在线观看| 欧美va亚洲va天堂va| 18禁免费av网站| 2018在线福利视频| 亚洲av色图18p| 夜夜操,天天操,狠狠操| 免费黄高清无码国产| 免费观看丰满少妇做受| 中文字幕乱码人妻电影| 国产高清在线观看1区2区| 沙月文乃人妻侵犯中文字幕在线| 啊用力插好舒服视频| 香蕉91一区二区三区| 宅男噜噜噜666免费观看| 99av国产精品欲麻豆| 99精品免费观看视频| 免费黄高清无码国产| 1024久久国产精品| 晚上一个人看操B片| 欧美日韩国产一区二区三区三州 | 久久这里有免费精品| 男人天堂色男人av| 老司机免费福利视频网| 精品suv一区二区69| 大尺度激情四射网站| 少妇高潮一区二区三区| 日韩视频一区二区免费观看| 欧美日韩一区二区电影在线观看| 成人动漫大肉棒插进去视频| 国产成人无码精品久久久电影| 亚洲 欧美 精品 激情 偷拍| 人妻无码色噜噜狠狠狠狠色| 天天躁日日躁狠狠躁av麻豆| 色婷婷六月亚洲综合香蕉| 91福利视频免费在线观看| 国产va在线观看精品| 国产va在线观看精品| 精品美女在线观看视频在线观看 | 亚洲一区二区三区av网站| 日本少妇在线视频大香蕉在线观看 | 国产精品亚洲在线观看| 欧洲亚洲欧美日韩综合| 亚洲一区二区三区偷拍女厕91| 非洲黑人一级特黄片| 免费69视频在线看| 二区中出在线观看老师| 亚洲人人妻一区二区三区| 亚洲福利天堂久久久久久 | 国产午夜亚洲精品麻豆| 国产亚洲天堂天天一区| 久久99久久99精品影院| 一区二区三区久久中文字幕| 色婷婷六月亚洲综合香蕉| 扒开腿挺进肉嫩小18禁视频| 天堂av狠狠操蜜桃| 欧美视频一区免费在线| 91精品国产观看免费| 亚洲视频乱码在线观看| 欧美精产国品一二三产品区别大吗| 亚洲一区二区三区久久受| 毛茸茸的大外阴中国视频| 肏插流水妹子在线乐播下载 | 人妻熟女中文字幕aⅴ在线| 免费十精品十国产网站| 黑人巨大精品欧美视频| 91免费放福利在线观看| 国产久久久精品毛片| 亚洲日本一区二区三区| 岛国黄色大片在线观看| 大黑人性xxxxbbbb| 91极品大一女神正在播放| 日日操夜夜撸天天干| 成人网18免费视频版国产| 大鸡吧插逼逼视频免费看 | 一本一本久久a久久精品综合不卡| 国产精品成人xxxx| 午夜免费观看精品视频| 一区二区三区四区中文| 999九九久久久精品| 青青草亚洲国产精品视频| 97a片免费在线观看| av日韩在线免费播放| 亚洲成人线上免费视频观看| 超污视频在线观看污污污| 欧美色婷婷综合在线| 久久久极品久久蜜桃| 丝袜长腿第一页在线| 天天日天天操天天摸天天舔| 蝴蝶伊人久久中文娱乐网| 亚洲一区二区三区在线高清| 无码国产精品一区二区高潮久久4| 9国产精品久久久久老师| 日本裸体熟妇区二区欧美| 色婷婷综合激情五月免费观看 | 国产极品精品免费视频| 日韩av免费观看一区| 国产熟妇人妻ⅹxxxx麻豆| 天干天天天色天天日天天射| 老鸭窝在线观看一区| 一区二区三区蜜臀在线| 99热久久这里只有精品8| 日本少妇人妻xxxxxhd| 日本又色又爽又黄又粗| 2020中文字幕在线播放| 国际av大片在线免费观看| 2022精品久久久久久中文字幕| 亚洲少妇高潮免费观看| 99一区二区在线观看| 农村胖女人操逼视频| 欧美日韩中文字幕欧美| 黄色资源视频网站日韩| 天天日天天天天天天天天天天| 日本高清在线不卡一区二区| japanese五十路熟女熟妇| 年轻的人妻被夫上司侵犯| 97精品成人一区二区三区| 九一传媒制片厂视频在线免费观看| 一区二区三区日韩久久| 国产精品视频欧美一区二区| 美女av色播在线播放| 日韩国产乱码中文字幕| 一区二区三区四区五区性感视频 | 成年人中文字幕在线观看| 国产亚洲精品品视频在线| 无码精品一区二区三区人| 91一区精品在线观看| 啪啪啪啪啪啪啪免费视频| 91福利视频免费在线观看| 欧美日韩中文字幕欧美| 欧美精品免费aaaaaa| 第一福利视频在线观看| 天天日天天日天天射天天干| 青青青青青手机视频| 青青青aaaa免费| 亚洲的电影一区二区三区 | 一区二区三区激情在线| 亚洲公开视频在线观看| 国产一区av澳门在线观看| 无忧传媒在线观看视频| 少妇深喉口爆吞精韩国| 欧美天堂av无线av欧美| 欧美交性又色又爽又黄麻豆| 天天干天天插天天谢| 国产成人无码精品久久久电影 | 天天做天天干天天操天天射| 9国产精品久久久久老师| 亚洲一区久久免费视频| 中文字幕之无码色多多| 精品高跟鞋丝袜一区二区| 男人天堂最新地址av| 亚洲成人国产综合一区| 91精品资源免费观看| 最近中文字幕国产在线| 国产97视频在线精品| 一区二区三区在线视频福利| 天天干天天操天天摸天天射| 激情五月婷婷免费视频| av在线shipin| 精品一线二线三线日本| 激情五月婷婷综合色啪| 天天日天天爽天天干| 和邻居少妇愉情中文字幕| 国产又粗又硬又大视频| 日韩欧美国产一区ab| 欧美少妇性一区二区三区| 国产av欧美精品高潮网站| 美女福利视频网址导航| 日韩欧美高清免费在线| 亚洲一区二区三区久久午夜 | 国产三级片久久久久久久 | 动色av一区二区三区| 女警官打开双腿沦为性奴| 黄网十四区丁香社区激情五月天| 国产福利在线视频一区| 欧美另类一区二区视频| 经典亚洲伊人第一页| 午夜91一区二区三区| 2020av天堂网在线观看| 青青青青青免费视频| 一本久久精品一区二区| 扒开腿挺进肉嫩小18禁视频| 亚洲 中文 自拍 另类 欧美| 国产综合高清在线观看| 青青尤物在线观看视频网站| 亚洲国产精品久久久久久6| 大香蕉日本伊人中文在线| 黄片大全在线观看观看| 视频 国产 精品 熟女 | av资源中文字幕在线观看| av亚洲中文天堂字幕网| 国产精品亚洲在线观看| 亚洲综合另类欧美久久| 中文字幕1卡1区2区3区| 欧美视频综合第一页| 国产精品久久久久久美女校花| 亚洲成人熟妇一区二区三区| 亚洲国产40页第21页| 亚洲欧美福利在线观看| 97国产福利小视频合集| 国产又大又黄免费观看| 亚洲精品国产久久久久久| 伊人综合免费在线视频| 大陆胖女人与丈夫操b国语高清| 成人亚洲国产综合精品| 清纯美女在线观看国产| 亚洲另类伦春色综合小| 亚洲欧美一区二区三区爱爱动图| 欧美精品一区二区三区xxxx| 欧美日韩国产一区二区三区三州| 色哟哟在线网站入口| 亚洲图片欧美校园春色| 国产精品国色综合久久| www天堂在线久久| 少妇一区二区三区久久久| 日韩亚国产欧美三级涩爱| 亚洲精品精品国产综合| 欧美韩国日本国产亚洲| 福利在线视频网址导航| 99视频精品全部15| av网址在线播放大全| 9久在线视频只有精品| 欧美日韩亚洲国产无线码| 日本韩国免费福利精品| 国产真实乱子伦a视频| 狠狠躁夜夜躁人人爽天天久天啪| 精品区一区二区三区四区人妻 | 精品区一区二区三区四区人妻| av中文字幕在线导航| 天天操天天干天天艹| 亚洲天堂精品福利成人av| 国产片免费观看在线观看| 欧美日韩精品永久免费网址| 精品区一区二区三区四区人妻| 狍和女人的王色毛片| 国产97视频在线精品| 初美沙希中文字幕在线| 亚洲av日韩精品久久久久久hd| 亚洲综合另类精品小说| 亚洲av男人天堂久久| 高潮喷水在线视频观看| 超碰在线中文字幕一区二区| 大香蕉伊人中文字幕| 制服丝袜在线人妻中文字幕| 婷婷久久久综合中文字幕| 搞黄色在线免费观看| 亚洲图片偷拍自拍区| 蜜桃久久久久久久人妻| 欧美韩国日本国产亚洲| 热99re69精品8在线播放| 熟女人妻三十路四十路人妻斩| 淫秽激情视频免费观看| 中文 成人 在线 视频| 日比视频老公慢点好舒服啊| 亚洲欧美清纯唯美另类 | 97青青青手机在线视频 | 成年人黄色片免费网站| 欧美黑人与人妻精品| 97人妻人人澡爽人人精品| 亚洲伊人久久精品影院一美女洗澡| 91破解版永久免费| 精品日产卡一卡二卡国色天香| 东京干手机福利视频| 天天干天天操天天爽天天摸| 久久综合老鸭窝色综合久久| 在线免费观看黄页视频| 人妻少妇精品久久久久久| 一区二区三区四区五区性感视频| 国产美女精品福利在线| 黑人大几巴狂插日本少妇| 91麻豆精品久久久久| 国内自拍第一页在线观看| 精品成人啪啪18免费蜜臀| 午夜场射精嗯嗯啊啊视频| 欧美xxx成人在线| 日本高清成人一区二区三区| 经典av尤物一区二区| 日本女大学生的黄色小视频| 9色精品视频在线观看| 桃色视频在线观看一区二区| 中文字幕一区二区自拍| 成人动漫大肉棒插进去视频| 99的爱精品免费视频| 成人综合亚洲欧美一区| 日韩精品二区一区久久| 免费男阳茎伸入女阳道视频| 亚洲无线观看国产高清在线| 欧美精品黑人性xxxx| 性欧美日本大妈母与子| 十八禁在线观看地址免费| 狠狠操操操操操操操操操| 人妻丰满熟妇综合网| 日本黄在免费看视频| 久久这里只有精彩视频免费| 欧美日韩人妻久久精品高清国产| yy6080国产在线视频| 中文字幕午夜免费福利视频| 天天射夜夜操综合网| 一区二区三区毛片国产一区| 毛片av在线免费看| 日韩精品二区一区久久| 中字幕人妻熟女人妻a62v网| 99av国产精品欲麻豆| 亚洲av日韩av网站| 天天干天天操天天摸天天射| 成人av电影免费版| 国产精品自偷自拍啪啪啪| 国产女人露脸高潮对白视频| 又黄又刺激的午夜小视频| 超级av免费观看一区二区三区| 日本少妇在线视频大香蕉在线观看 | 这里只有精品双飞在线播放| 精品美女福利在线观看| 亚洲中文字幕国产日韩| 2022国产综合在线干| 非洲黑人一级特黄片| 青青青爽视频在线播放| 偷拍自拍福利视频在线观看| 精品亚洲在线免费观看| 亚洲变态另类色图天堂网| 精品亚洲中文字幕av| 日韩a级黄色小视频| 班长撕开乳罩揉我胸好爽| 日韩亚洲高清在线观看| 中文字幕中文字幕人妻| 久久久久久性虐视频| 天天躁夜夜躁日日躁a麻豆| 91精品国产黑色丝袜| 久碰精品少妇中文字幕av| 亚洲av自拍偷拍综合| 久久久久久97三级| 欧美黑人巨大性xxxxx猛交| 欧美精品黑人性xxxx| 93视频一区二区三区| 欧美日韩亚洲国产无线码| 国产又色又刺激在线视频 | 91老熟女连续高潮对白| 中文字幕在线第一页成人| 粉嫩欧美美人妻小视频| 国产a级毛久久久久精品| 干逼又爽又黄又免费的视频| 98精产国品一二三产区区别| 在线观看亚洲人成免费网址| 黄片三级三级三级在线观看| 天天干天天搞天天摸| 538精品在线观看视频| 久久久噜噜噜久久熟女av| 亚洲av无乱一区二区三区性色| 国产免费高清视频视频| av在线资源中文字幕| 人妻3p真实偷拍一二区| 国产在线自在拍91国语自产精品| 久久免费看少妇高潮完整版| 大黑人性xxxxbbbb| 在线观看视频一区麻豆| 国产久久久精品毛片| 自拍偷拍日韩欧美一区二区| 99久久中文字幕一本人| 亚洲精品亚洲人成在线导航| 中文字幕av一区在线观看| 久久久久久久久久久久久97| 男人天堂最新地址av| 国产大鸡巴大鸡巴操小骚逼小骚逼| 日韩午夜福利精品试看| 欧美区一区二区三视频| 亚洲国产成人av在线一区| 超pen在线观看视频公开97| 国产又粗又黄又硬又爽| 激情啪啪啪啪一区二区三区| 国产chinesehd精品麻豆| 国产视频在线视频播放| 好太好爽好想要免费| 大胸性感美女羞爽操逼毛片| 18禁免费av网站| 大陆av手机在线观看| 亚洲超碰97人人做人人爱| 中国黄色av一级片| 欧洲精品第一页欧洲精品亚洲| 免费看国产av网站| 国产精品三级三级三级| 国产欧美日韩第三页| 青青青青青青青青青国产精品视频| 精品一区二区三四区| 日本a级视频老女人| 日韩精品二区一区久久| 亚洲国产精品久久久久蜜桃| 国产欧美日韩第三页| 国产一区二区神马久久| 青青青国产片免费观看视频| 欧美日本在线观看一区二区| 2020av天堂网在线观看| 动漫美女的小穴视频| 在线制服丝袜中文字幕| 美女福利视频导航网站 | 一区二区三区久久中文字幕| 亚洲欧美清纯唯美另类| heyzo蜜桃熟女人妻| 夜夜骑夜夜操夜夜奸| 日韩av有码一区二区三区4| 绯色av蜜臀vs少妇| 国产亚洲精品欧洲在线观看| 521精品视频在线观看| 天天操天天干天天日狠狠插| 日韩美女搞黄视频免费| 国产片免费观看在线观看| 老司机你懂得福利视频| 少妇人妻真实精品视频| 韩国爱爱视频中文字幕| 欧美黄片精彩在线免费观看| 91精品国产91久久自产久强| 国产日韩精品免费在线| 国产一级精品综合av| 99国内小视频在现欢看| 亚洲国产精品久久久久久6| 亚洲av色香蕉一区二区三区| 亚洲av一妻不如妾| 国产高潮无码喷水AV片在线观看| 一区二区三区久久中文字幕| 毛茸茸的大外阴中国视频| 精品黑人一区二区三区久久国产| 特一级特级黄色网片| 自拍偷区二区三区麻豆| 国产精品国产三级国产精东| 亚洲熟女综合色一区二区三区四区| 成人sm视频在线观看| 欧美成人黄片一区二区三区| 熟妇一区二区三区高清版| 日韩二区视频一线天婷婷五| 在线观看免费视频色97| 亚洲激情,偷拍视频| 老鸭窝日韩精品视频观看| 日本性感美女视频网站| 福利视频广场一区二区| 国产精品久久久黄网站| 91国产资源在线视频| 丰满的子国产在线观看| 老鸭窝日韩精品视频观看| 国产日韩欧美视频在线导航| 沙月文乃人妻侵犯中文字幕在线| 操的小逼流水的文章| 亚洲av日韩av网站| 欧美交性又色又爽又黄麻豆| 超碰在线观看免费在线观看| 丰满的继坶3中文在线观看| 在线免费观看日本伦理| 青青青青视频在线播放| 久久久久91精品推荐99| 人妻少妇av在线观看| 亚洲欧美人精品高清| 经典av尤物一区二区| 天天色天天爱天天爽| 99精品国产aⅴ在线观看| 黄色在线观看免费观看在线| 国产三级影院在线观看| 2012中文字幕在线高清| 成人福利视频免费在线| chinese国产盗摄一区二区| 亚洲最大黄了色网站| 97超碰国语国产97超碰| 丝袜亚洲另类欧美变态| 日韩二区视频一线天婷婷五| 在线免费观看黄页视频| 老熟妇xxxhd老熟女| 干逼又爽又黄又免费的视频| 丁香花免费在线观看中文字幕| 国产精品国色综合久久| 美女张开腿让男生操在线看| 天天操天天操天天碰| 欧美精品黑人性xxxx| 精品亚洲在线免费观看| 硬鸡巴动态操女人逼视频| 精品高潮呻吟久久av| 亚洲久久午夜av一区二区| 黄色三级网站免费下载| 色爱av一区二区三区| 少妇高潮一区二区三区| 欧美老鸡巴日小嫩逼| 青青青国产免费视频| 亚洲免费在线视频网站| 在线免费观看国产精品黄色| 夜女神免费福利视频| 欧美精品资源在线观看| 欧美性感尤物人妻在线免费看| 巨乳人妻日下部加奈被邻居中出 | 亚洲一级av大片免费观看| 免费高清自慰一区二区三区网站 | 55夜色66夜色国产精品站| 97人妻色免费视频| 九色精品视频在线播放| 国产美女一区在线观看| 精品一区二区亚洲欧美| 欧美精产国品一二三产品区别大吗| 欧美另类重口味极品在线观看| 丝袜国产专区在线观看| 十八禁在线观看地址免费| 国产麻豆乱子伦午夜视频观看| 激情五月婷婷免费视频| 97人人模人人爽人人喊| 亚洲国产精品中文字幕网站| 成人av中文字幕一区| 在线播放国产黄色av| 亚洲视频在线观看高清| 午夜国产福利在线观看| 男女啪啪啪啪啪的网站| AV天堂一区二区免费试看| 2021最新热播中文字幕| 欧美3p在线观看一区二区三区| 日本五十路熟新垣里子| 中英文字幕av一区| 国产露脸对白在线观看| 成人免费公开视频无毒| 日本美女成人在线视频| 色吉吉影音天天干天天操 | 国产熟妇人妻ⅹxxxx麻豆| 亚洲中文字幕综合小综合| 51精品视频免费在线观看| 人妻少妇亚洲一区二区| caoporm超碰国产| 久久99久久99精品影院| 91av中文视频在线| 日韩影片一区二区三区不卡免费 | 51精品视频免费在线观看| 91久久精品色伊人6882| 超污视频在线观看污污污| 久久99久久99精品影院| 熟女在线视频一区二区三区| weyvv5国产成人精品的视频| 年轻的人妻被夫上司侵犯| 岛国免费大片在线观看| 日视频免费在线观看| 伊人成人在线综合网| 欧美日韩中文字幕欧美| 懂色av之国产精品| 人人妻人人爱人人草| 欧美精产国品一二三产品价格| 中文字幕日韩精品日本| 最新97国产在线视频| 国产av欧美精品高潮网站| mm131美女午夜爽爽爽| 黑人大几巴狂插日本少妇| 在线观看免费视频网| 亚洲最大黄 嗯色 操 啊| 美女张开两腿让男人桶av| 亚洲人妻视频在线网| 又黄又刺激的午夜小视频| 人妻久久久精品69系列| 久久国产精品精品美女| 最近中文2019年在线看| 91av中文视频在线| 快插进小逼里大鸡吧视频| 国产黄色高清资源在线免费观看| 成年人的在线免费视频| 丰满的子国产在线观看| 亚洲另类伦春色综合小| 午夜久久久久久久精品熟女| 动色av一区二区三区| 在线免费视频 自拍| 国产自拍黄片在线观看| 日韩av熟妇在线观看| 大学生A级毛片免费视频| 爆乳骚货内射骚货内射在线| 亚洲一区二区三区久久午夜| 亚洲的电影一区二区三区| 深夜男人福利在线观看| 在线免费观看av日韩| 超碰在线观看免费在线观看| 日韩欧美中文国产在线| 国产变态另类在线观看| 国产性色生活片毛片春晓精品 | 伊人网中文字幕在线视频| 亚洲成人线上免费视频观看| rct470中文字幕在线| 亚洲一区二区三区久久受 | 亚洲变态另类色图天堂网| 色哟哟国产精品入口| 国产日韩一区二区在线看| 中文字幕高清免费在线人妻| 日本少妇人妻xxxxxhd| 日本免费视频午夜福利视频| 日本最新一二三区不卡在线 | 2022国产综合在线干| 欧美精品黑人性xxxx| 在线免费观看日本片| 丰满的子国产在线观看| 熟女在线视频一区二区三区| 天堂av在线播放免费| 亚洲午夜电影在线观看| 桃色视频在线观看一区二区| 夜夜嗨av一区二区三区中文字幕| 欧美亚洲一二三区蜜臀| 在线免费观看欧美小视频| 黄色片年轻人在线观看| 99精品国产免费久久| 2020久久躁狠狠躁夜夜躁| 中文字幕中文字幕 亚洲国产| 国产欧美精品不卡在线| 日韩激情文学在线视频| 人妻熟女中文字幕aⅴ在线| 国产精品中文av在线播放| 啪啪啪18禁一区二区三区| 久久www免费人成一看片| 国产在线一区二区三区麻酥酥| 91‖亚洲‖国产熟女| 爱有来生高清在线中文字幕| 一区二区三区视频,福利一区二区| 激情小视频国产在线 | 国产午夜亚洲精品麻豆| 婷婷久久久综合中文字幕| 动漫黑丝美女的鸡巴| 亚洲码av无色中文| 丝袜肉丝一区二区三区四区在线| 538精品在线观看视频| 欧美少妇性一区二区三区| 9色精品视频在线观看| 在线观看的a站 最新| 老司机在线精品福利视频| 国产剧情演绎系列丝袜高跟| 大学生A级毛片免费视频| 天天日天天干天天干天天日| 青娱乐最新视频在线| 国产久久久精品毛片| 亚洲欧美清纯唯美另类 | 超碰97人人澡人人| 亚洲 自拍 色综合图| 日韩一区二区电国产精品| 中文字幕1卡1区2区3区| 女同性ⅹxx女同hd| av高潮迭起在线观看| 女生自摸在线观看一区二区三区 | 国产美女午夜福利久久| 韩国女主播精品视频网站| 免费在线看的黄网站| 亚洲激情av一区二区| 欧美 亚洲 另类综合| 激情小视频国产在线| 日韩精品中文字幕福利| 在线观看av亚洲情色| 国产无遮挡裸体免费直播视频| 天天躁夜夜躁日日躁a麻豆| 大骚逼91抽插出水视频| 最近中文字幕国产在线| 久久久久久cao我的性感人妻| 亚洲最大黄 嗯色 操 啊| 亚洲一区二区三区在线高清| 97青青青手机在线视频| 日辽宁老肥女在线观看视频| 天天色天天舔天天射天天爽| 日本中文字幕一二区视频| 香蕉片在线观看av| 91在线免费观看成人| 少妇露脸深喉口爆吞精| 538精品在线观看视频| 在线观看免费av网址大全| 最新91精品视频在线| 国产视频一区在线观看| 老鸭窝日韩精品视频观看| 午夜精品亚洲精品五月色| 天天操天天干天天插| 美女av色播在线播放| 福利视频网久久91| 国产午夜无码福利在线看| 中文字幕人妻被公上司喝醉在线 | 亚洲成人线上免费视频观看| 天天日天天日天天擦| 特一级特级黄色网片| 成人高清在线观看视频| 国产aⅴ一线在线观看| 日本午夜福利免费视频| 青青青青青青青在线播放视频| 香港一级特黄大片在线播放| 国产麻豆国语对白露脸剧情| 啪啪啪啪啪啪啪啪啪啪黄色| 一区二区三区麻豆福利视频| 2022国产精品视频| 成人蜜桃美臀九一一区二区三区| 成人区人妻精品一区二视频| 99精品亚洲av无码国产另类| 成人伊人精品色xxxx视频| 亚洲 清纯 国产com| 人人妻人人澡人人爽人人dvl| 不卡一不卡二不卡三| 国产精品人久久久久久| 亚洲av日韩精品久久久久久hd| 自拍偷拍vs一区二区三区| 国产精品女邻居小骚货| 天天摸天天亲天天舔天天操天天爽 | 97超碰免费在线视频| 国产午夜福利av导航| 亚洲熟女久久久36d| 五月天久久激情视频| 视频啪啪啪免费观看| 好了av中文字幕在线| 亚洲成人熟妇一区二区三区| 人妻无码色噜噜狠狠狠狠色 | 成人免费公开视频无毒| 久久久久久cao我的性感人妻| 涩爱综合久久五月蜜臀| 亚洲成人激情av在线| 在线观看视频污一区| 青青尤物在线观看视频网站| 天天干天天操天天插天天日| 91老熟女连续高潮对白| 日本性感美女写真视频| 51国产成人精品视频| 成人高清在线观看视频| okirakuhuhu在线观看| 大肉大捧一进一出好爽在线视频| 午夜精品福利91av| 日韩影片一区二区三区不卡免费| 夫妻在线观看视频91| 中文字幕一区二区亚洲一区| 欧洲精品第一页欧洲精品亚洲| 日本在线不卡免费视频| 青青社区2国产视频| 九一传媒制片厂视频在线免费观看| 亚洲码av无色中文| 亚洲激情唯美亚洲激情图片| 国语对白xxxx乱大交| 精品av国产一区二区三区四区| 2022国产综合在线干| 国产精品亚洲а∨天堂免| 亚洲中文精品人人免费| 国产精品国产三级国产午| 亚洲精品乱码久久久本| 精品亚洲国产中文自在线| asmr福利视频在线观看| 亚洲中文字字幕乱码| 91久久综合男人天堂| 大香蕉大香蕉在线看| 人人人妻人人澡人人| 天天躁夜夜躁日日躁a麻豆| 老司机深夜免费福利视频在线观看| 91亚洲国产成人精品性色| av高潮迭起在线观看| 337p日本大胆欧美人| 91精品资源免费观看| 99一区二区在线观看| 视频 国产 精品 熟女 | 女同性ⅹxx女同hd| 亚洲图库另类图片区| 漂亮 人妻被中出中文| 国产精品人久久久久久| 亚洲精品久久视频婷婷| 五月天久久激情视频| 国产女人露脸高潮对白视频| 99久久超碰人妻国产| 毛茸茸的大外阴中国视频| 色婷婷精品大在线观看| 一区二区三区四区视频| 动漫美女的小穴视频| 成人高清在线观看视频| 性生活第二下硬不起来| 亚洲精品精品国产综合| 免费观看国产综合视频| 日本在线一区二区不卡视频| 亚洲欧美综合另类13p| 黄色三级网站免费下载| 黄网十四区丁香社区激情五月天 | 中文字幕在线观看国产片| 97精品视频在线观看| 懂色av蜜桃a v| 大胆亚洲av日韩av| 成人30分钟免费视频| 精品美女福利在线观看| 狠狠鲁狠狠操天天晚上干干| 夏目彩春在线中文字幕| 中文字幕1卡1区2区3区| jiuse91九色视频| 91成人在线观看免费视频| 亚洲第17页国产精品| 国产一级精品综合av| 熟女俱乐部一二三区| 又粗又硬又猛又爽又黄的| 熟女人妻三十路四十路人妻斩| 538精品在线观看视频| 国产熟妇人妻ⅹxxxx麻豆| 韩国一级特黄大片做受| 日本高清撒尿pissing| 国产黄色片在线收看| 18禁美女羞羞免费网站| 少妇人妻二三区视频| 黑人乱偷人妻中文字幕| 99久久久无码国产精品性出奶水| 福利一二三在线视频观看| 区一区二区三国产中文字幕| 激情国产小视频在线| 天天操天天操天天碰| 午夜在线一区二区免费| 欧美成人综合视频一区二区| 日本丰满熟妇大屁股久久| 午夜久久久久久久99| 亚洲中文精品人人免费| 亚洲欧美在线视频第一页| 99国产精品窥熟女精品| 亚洲 色图 偷拍 欧美| 五月激情婷婷久久综合网| 中文字幕高清免费在线人妻| 免费啪啪啪在线观看视频| 丝袜美腿欧美另类 中文字幕| 1000部国产精品成人观看视频 | 亚洲午夜电影之麻豆| 亚洲狠狠婷婷综合久久app| 夜夜嗨av一区二区三区中文字幕| 亚洲熟色妇av日韩熟色妇在线| 久久精品36亚洲精品束缚| 亚洲av男人天堂久久| 最新中文字幕免费视频| 午夜精品一区二区三区更新| 老司机99精品视频在线观看 | 97香蕉碰碰人妻国产樱花| av天堂资源最新版在线看| 日本女大学生的黄色小视频| 亚洲成人国产综合一区| 亚洲天堂有码中文字幕视频| 亚洲第一黄色在线观看| 亚洲综合图片20p| 欧美激情精品在线观看| 亚洲成人免费看电影| 日本高清撒尿pissing| 日日操夜夜撸天天干| 自拍偷拍日韩欧美一区二区| 51国产成人精品视频| 91人妻人人做人人爽在线| 91精品国产91青青碰| 亚洲男人的天堂a在线| 婷婷色国产黑丝少妇勾搭AV | 青草青永久在线视频18| caoporm超碰国产| 视频 一区二区在线观看| 中文字幕在线一区精品| 久草电影免费在线观看| 亚洲欧美激情人妻偷拍| 日本精品视频不卡一二三| 色av色婷婷人妻久久久精品高清| 日韩人妻在线视频免费| 天天日天天干天天干天天日| 夜夜嗨av一区二区三区中文字幕| 青娱乐在线免费视频盛宴| 自拍偷拍日韩欧美亚洲| 97瑟瑟超碰在线香蕉| 成人高清在线观看视频| 午夜激情久久不卡一区二区| 久久香蕉国产免费天天| 亚洲一区二区三区av网站| japanese日本熟妇另类| 国产视频网站一区二区三区| 色呦呦视频在线观看视频| 精品一区二区三四区| 91精品国产综合久久久蜜| 老熟妇xxxhd老熟女| 久久国产精品精品美女| brazzers欧熟精品系列| 黄色视频在线观看高清无码| 欧美性感尤物人妻在线免费看| 爱有来生高清在线中文字幕| 懂色av蜜桃a v| 瑟瑟视频在线观看免费视频| 黄色男人的天堂视频| 适合午夜一个人看的视频| 欧美爆乳肉感大码在线观看| 日本又色又爽又黄又粗| 在线观看成人国产电影| 久久尻中国美女视频| 亚洲av色图18p| 亚洲熟女女同志女同| 最新91九色国产在线观看| 天天日天天做天天日天天做| 久久久久久国产精品| 天天操天天干天天日狠狠插| 久久精品视频一区二区三区四区| 青草亚洲视频在线观看| 黄色片黄色片wyaa| 中文字幕 人妻精品| 九九视频在线精品播放| 色天天天天射天天舔| 国产黄色a级三级三级三级| 亚洲 人妻 激情 中文| 18禁无翼鸟成人在线| 欧美中文字幕一区最新网址| 日本福利午夜电影在线观看| 国产精品福利小视频a| 国产女人露脸高潮对白视频| 欧美成人小视频在线免费看| 午夜精品福利91av| 干逼又爽又黄又免费的视频| 又粗又长 明星操逼小视频| 天天日天天日天天射天天干| 欧美另类重口味极品在线观看| av高潮迭起在线观看| 专门看国产熟妇的网站| 亚洲av日韩精品久久久| 欧美怡红院视频在线观看| 亚洲成人情色电影在线观看| 男人靠女人的逼视频| 国产在线观看黄色视频| 美女日逼视频免费观看| 大肉大捧一进一出好爽在线视频| 成人蜜桃美臀九一一区二区三区| 玖玖一区二区在线观看| 97年大学生大白天操逼| 桃色视频在线观看一区二区| 青青色国产视频在线| 国产精品久久久黄网站| 一区二区三区四区五区性感视频| 中文字日产幕乱六区蜜桃| 天天日天天干天天要| 亚洲1069综合男同| 91极品大一女神正在播放| 婷婷久久一区二区字幕网址你懂得| 精内国产乱码久久久久久| 40道精品招牌菜特色| 久久久久久久精品成人热| 亚洲综合另类精品小说| 丝袜肉丝一区二区三区四区在线看| 亚洲欧洲av天堂综合| 亚洲av自拍天堂网| 成人乱码一区二区三区av| 91极品新人『兔兔』精品新作| 在线播放 日韩 av| 欧美在线一二三视频| jiuse91九色视频| 久草免费人妻视频在线| 天天摸天天亲天天舔天天操天天爽| 韩国黄色一级二级三级| 直接能看的国产av| 美女福利写真在线观看视频| 成人伊人精品色xxxx视频| 欧美黑人与人妻精品| 大肉大捧一进一出好爽在线视频| 91精品一区二区三区站长推荐| 熟女视频一区,二区,三区 | 日日爽天天干夜夜操| 午夜在线精品偷拍一区二| 国产一区二区在线欧美| 国产欧美日韩第三页| 成人蜜桃美臀九一一区二区三区| 人妻久久久精品69系列| 91麻豆精品91久久久久同性| 精品国产高潮中文字幕| 一区二区三区久久中文字幕| 超级福利视频在线观看| 日本18禁久久久久久| 小穴多水久久精品免费看| 男人天堂色男人av| 亚洲精品国产久久久久久| 婷婷久久一区二区字幕网址你懂得| 夜夜嗨av蜜臀av| 亚洲一区二区三区久久受| 青青青aaaa免费| 国产在线观看免费人成短视频| 天天色天天操天天透| 亚洲人妻30pwc| 亚洲av可乐操首页| 岛国黄色大片在线观看| 天天干天天操天天玩天天射| 国产普通话插插视频| 亚洲人妻国产精品综合| www天堂在线久久| 久久这里只有精品热视频| 麻豆性色视频在线观看| 涩涩的视频在线观看视频| 无码国产精品一区二区高潮久久4| 在线观看亚洲人成免费网址| 天天日天天爽天天干| 九色视频在线观看免费| av欧美网站在线观看| 91麻豆精品久久久久| 天天射夜夜操综合网| 可以在线观看的av中文字幕| 大鸡吧插逼逼视频免费看| 欧美视频一区免费在线| 亚洲av自拍偷拍综合|