国产无遮挡裸体免费直播视频,久久精品国产蜜臀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服務器了。這個服務器的功能可以進行完全個性化定制。如果您有好的建議,也可以留言給我,謝謝。

相關文章

最新評論

午夜美女少妇福利视频| 热99re69精品8在线播放| 亚洲熟妇久久无码精品| 一区二区在线视频中文字幕| 亚洲成人av一区久久| 18禁污污污app下载| 精品老妇女久久9g国产| 一级黄色片夫妻性生活| 55夜色66夜色国产精品站| 大香蕉大香蕉在线有码 av| 中文亚洲欧美日韩无线码| 91色网站免费在线观看| 99热久久这里只有精品| 久久久久五月天丁香社区| www日韩a级s片av| 黄色成人在线中文字幕| 久久人人做人人妻人人玩精品vr| 日本少妇人妻xxxxxhd| 热久久只有这里有精品| 9久在线视频只有精品| 福利午夜视频在线合集| 亚洲 中文 自拍 另类 欧美| 成人国产影院在线观看| 操操网操操伊剧情片中文字幕网| 另类av十亚洲av| 免费黄高清无码国产| 中文字幕中文字幕 亚洲国产| rct470中文字幕在线| 欧洲亚洲欧美日韩综合| 只有精品亚洲视频在线观看| 把腿张开让我插进去视频| 日视频免费在线观看| 亚洲一级特黄特黄黄色录像片| 直接能看的国产av| 超pen在线观看视频公开97| 久久久久久久久久性潮| 精品一区二区三四区| 日日夜夜大香蕉伊人| 精品久久婷婷免费视频| 自拍偷拍亚洲欧美在线视频| av在线观看网址av| 激情啪啪啪啪一区二区三区| av中文字幕在线观看第三页| 精品少妇一二三视频在线| 国产污污污污网站在线| 男人操女人的逼免费视频| 大香蕉伊人中文字幕| 国产综合精品久久久久蜜臀| 日韩美女搞黄视频免费| 香蕉aⅴ一区二区三区| 高潮喷水在线视频观看| 美女张开两腿让男人桶av| 无套猛戳丰满少妇人妻| 91社福利《在线观看| 午夜精品一区二区三区福利视频| 人人妻人人爽人人澡人人精品| 特级欧美插插插插插bbbbb| 国产精品亚洲а∨天堂免| 久久久制服丝袜中文字幕| 亚洲Av无码国产综合色区| 人妻少妇av在线观看| 天天日天天日天天射天天干| gogo国模私拍视频| 99精品视频在线观看免费播放| 日本一道二三区视频久久| 亚洲国产精品黑丝美女| 亚洲 色图 偷拍 欧美| 亚洲va欧美va人人爽3p| 啊啊啊想要被插进去视频| 老鸭窝日韩精品视频观看| 天天干天天日天天干天天操| 成人蜜桃美臀九一一区二区三区| 天美传媒mv视频在线观看| 亚洲欧美另类自拍偷拍色图| 97精品综合久久在线| 自拍偷拍一区二区三区图片| 亚洲蜜臀av一区二区三区九色| 亚洲粉嫩av一区二区三区| 蜜臀av久久久久蜜臀av麻豆| 日本阿v视频在线免费观看| 老司机福利精品免费视频一区二区 | 亚洲免费国产在线日韩| 亚洲av色图18p| 成人24小时免费视频| 最新日韩av传媒在线| 国产一区二区欧美三区| 国产精品福利小视频a| 视频 一区二区在线观看| 国产精品久久综合久久| 青青青青爽手机在线| 蜜桃精品久久久一区二区| 亚洲男人的天堂a在线| 欧美日韩不卡一区不区二区| 含骚鸡巴玩逼逼视频| 亚洲伊人久久精品影院一美女洗澡| 免费观看理论片完整版| 香蕉av影视在线观看| 日本韩国免费福利精品| 欧美黑人巨大性xxxxx猛交| 麻豆性色视频在线观看| 欧美亚洲牲夜夜综合久久| 大香蕉大香蕉大香蕉大香蕉大香蕉| 男人操女人逼逼视频网站| aiss午夜免费视频| 91国产在线免费播放| lutube在线成人免费看| 少妇深喉口爆吞精韩国| 密臀av一区在线观看| 欧洲精品第一页欧洲精品亚洲 | 男女之间激情网午夜在线| 天天干天天操天天插天天日| 五十路在线观看完整版| 国产视频一区在线观看| 都市激情校园春色狠狠| 最近的中文字幕在线mv视频| 日本韩国亚洲综合日韩欧美国产| 亚洲欧美福利在线观看| 好男人视频在线免费观看网站| 美洲精品一二三产区区别| 亚洲av一妻不如妾| 超黄超污网站在线观看| www,久久久,com| 国产1区,2区,3区| 国产三级精品三级在线不卡| 99久久99一区二区三区| 成人亚洲国产综合精品| 自拍偷拍亚洲另类色图| 护士特殊服务久久久久久久| 一本久久精品一区二区| 五月天中文字幕内射| 婷婷综合亚洲爱久久| 欧美日韩情色在线观看| 国产日韩欧美视频在线导航| 91精品国产高清自在线看香蕉网| 日韩欧美亚洲熟女人妻| 韩国一级特黄大片做受| 日本免费一级黄色录像| 亚洲欧美色一区二区| 亚洲欧美久久久久久久久| 美女视频福利免费看| 色爱av一区二区三区| 75国产综合在线视频| 国产精品三级三级三级| 在线免费观看欧美小视频| 久精品人妻一区二区三区 | 天天干天天操天天爽天天摸| 亚洲精品乱码久久久久久密桃明| 日韩精品电影亚洲一区| 免费黄高清无码国产| 国产又粗又硬又猛的毛片视频 | 好太好爽好想要免费| 香蕉91一区二区三区| 五十路丰满人妻熟妇| 精品亚洲中文字幕av| 亚洲激情唯美亚洲激情图片| 久久久久久性虐视频| 黄色视频在线观看高清无码| 黄色三级网站免费下载| 精品视频一区二区三区四区五区| 性色av一区二区三区久久久| 骚逼被大屌狂草视频免费看| 夜夜骑夜夜操夜夜奸| 最近的中文字幕在线mv视频| 色综合色综合色综合色| 日韩美女搞黄视频免费| 18禁美女黄网站色大片下载| 2022国产精品视频| av网址国产在线观看| 宅男噜噜噜666国产| 天天干天天日天天谢综合156| 老司机在线精品福利视频| 亚洲欧美国产麻豆综合| 亚洲精品国偷自产在线观看蜜桃| 欧美特级特黄a大片免费| 漂亮 人妻被中出中文| 一区二区视频视频视频| 中文字幕人妻熟女在线电影| 欧美韩国日本国产亚洲| 日本欧美视频在线观看三区| 任我爽精品视频在线播放| 91香蕉成人app下载| 视频久久久久久久人妻| 在线视频免费观看网| 一二三中文乱码亚洲乱码one | 男人操女人逼逼视频网站| 国产三级片久久久久久久| 一区二区三区四区视频在线播放| 亚洲人成精品久久久久久久| 91久久国产成人免费网站| yy96视频在线观看| 无码日韩人妻精品久久| 欧美怡红院视频在线观看| 哥哥姐姐综合激情小说| 亚洲视频乱码在线观看| 无码日韩人妻精品久久| 国产一区av澳门在线观看| 中文字幕日韩精品日本| 中文字幕在线一区精品| 老司机深夜免费福利视频在线观看| 亚洲福利天堂久久久久久| 91大神福利视频网| 国产亚洲视频在线观看| 99一区二区在线观看| 日本黄色三级高清视频| 青青伊人一精品视频| 风流唐伯虎电视剧在线观看 | 97人人妻人人澡人人爽人人精品| 午夜精品亚洲精品五月色| 最新日韩av传媒在线| 精品一区二区三区在线观看| 777奇米久久精品一区| 99re6热在线精品| 91快播视频在线观看| 香港三日本三韩国三欧美三级| 免费黄页网站4188| 欧美特色aaa大片| 班长撕开乳罩揉我胸好爽| 中字幕人妻熟女人妻a62v网| 欧美另类z0z变态| 亚洲欧美久久久久久久久| 欧美怡红院视频在线观看| 任你操视频免费在线观看| 天天射,天天操,天天说| 天干天天天色天天日天天射| 欲乱人妻少妇在线视频裸| 日本www中文字幕| 欧美 亚洲 另类综合| 成年人午夜黄片视频资源| 91九色porny蝌蚪国产成人| 少妇ww搡性bbb91| 天天日天天敢天天干| 在线免费观看视频一二区| 在线观看免费视频网| 中文字幕亚洲中文字幕| 五月天中文字幕内射| 国产高清在线在线视频| 99精品亚洲av无码国产另类| 亚洲精品福利网站图片| 水蜜桃一区二区三区在线观看视频| 成熟熟女国产精品一区| 亚洲码av无色中文| 亚洲人妻av毛片在线| 日本午夜爽爽爽爽爽视频在线观看| 99国内小视频在现欢看| 五十路熟女av天堂| 久久这里只有精品热视频| 国产一级精品综合av| 午夜青青草原网在线观看| 一区二区三区国产精选在线播放 | 久久久久久久久久久免费女人| 天天干天天爱天天色| 日本一区精品视频在线观看| 5528327男人天堂| 日本熟妇一区二区x x| 欧美久久久久久三级网| 亚洲成人av一区久久| 最后99天全集在线观看| 日韩人妻丝袜中文字幕| 男人的天堂av日韩亚洲| 综合激情网激情五月天| 日本性感美女三级视频| 午夜久久久久久久99| 护士特殊服务久久久久久久| 好男人视频在线免费观看网站| 美女吃鸡巴操逼高潮视频| 青青青青爽手机在线| 欧美精品激情在线最新观看视频| 久久久久91精品推荐99| 91精品免费久久久久久| 国产真实乱子伦a视频| 伊人情人综合成人久久网小说| 成人国产影院在线观看| 熟女少妇激情五十路| 日韩熟女av天堂系列| 成人精品在线观看视频| 亚洲国产精品黑丝美女| 国产精品自拍偷拍a| 国产精品sm调教视频| 男人在床上插女人视频| 揄拍成人国产精品免费看视频| 人妻无码中文字幕专区| 又粗又长 明星操逼小视频| 蜜桃臀av蜜桃臀av| 日本阿v视频在线免费观看| 国产使劲操在线播放| 青娱乐极品视频青青草| 成人精品视频99第一页| 热久久只有这里有精品| 丝袜美腿视频诱惑亚洲无| 色花堂在线av中文字幕九九 | 91桃色成人网络在线观看| 热久久只有这里有精品| 亚洲精品午夜aaa久久| 日本成人一区二区不卡免费在线| 99精品视频在线观看婷婷| 中文字幕+中文字幕| 国产综合高清在线观看| 国产一区av澳门在线观看| 亚洲推理片免费看网站| 亚洲国产欧美一区二区三区久久| 粉嫩av懂色av蜜臀av| 北条麻妃肉色丝袜视频| 久久久久91精品推荐99| 天天日天天做天天日天天做| 欧美日韩人妻久久精品高清国产| 青青草在观免费国产精品| 在线观看日韩激情视频| 国产精品久久久久久久精品视频| 漂亮 人妻被中出中文| 激情五月婷婷综合色啪| 激情综合治理六月婷婷| 姐姐的朋友2在线观看中文字幕| 99精品国产自在现线观看| 亚洲午夜在线视频福利| 久久久制服丝袜中文字幕| 亚国产成人精品久久久| 小泽玛利亚视频在线观看| 插小穴高清无码中文字幕| 蜜桃专区一区二区在线观看| 亚洲人一区二区中文字幕| 又粗又硬又猛又黄免费30| 熟女少妇激情五十路| 成人免费公开视频无毒| 亚洲 欧美 精品 激情 偷拍| 黄色中文字幕在线播放| 人妻素人精油按摩中出| 天天日天天日天天擦| 中文字幕中文字幕人妻| 亚洲天堂有码中文字幕视频| 自拍偷拍,中文字幕| 不卡一区一区三区在线| 亚洲va天堂va国产va久| 国产精品一区二区av国| 99热这里只有国产精品6| 日本www中文字幕| 日辽宁老肥女在线观看视频| 一区国内二区日韩三区欧美| 福利午夜视频在线合集| 97人妻无码AV碰碰视频| 99久久激情婷婷综合五月天| 美女张开腿让男生操在线看| 人妻无码中文字幕专区| 天天日天天干天天干天天日| 66久久久久久久久久久| 成人福利视频免费在线| 日韩欧美国产一区不卡| 国产午夜亚洲精品麻豆| 大香蕉福利在线观看| 国产精品熟女久久久久浪潮| 国产精品久久久久国产三级试频| 国产污污污污网站在线| 黄色中文字幕在线播放| 天天干天天爱天天色| 日本福利午夜电影在线观看| 亚洲免费va在线播放| 亚洲1卡2卡三卡4卡在线观看| 黄色片年轻人在线观看| 天天色天天舔天天射天天爽| 国产亚洲成人免费在线观看| 亚洲一区二区久久久人妻| 自拍偷拍日韩欧美亚洲| 亚洲高清国产自产av| 99久久成人日韩欧美精品| 福利在线视频网址导航| 一本一本久久a久久精品综合不卡| 欧亚日韩一区二区三区观看视频| 亚洲免费av在线视频| 日韩人妻xxxxx| 老司机99精品视频在线观看| 国产精彩对白一区二区三区| 亚洲精品久久综合久| 日韩美女搞黄视频免费| 欧美精品中文字幕久久二区| 这里只有精品双飞在线播放| 午夜dv内射一区区| 日本韩国亚洲综合日韩欧美国产| 2020韩国午夜女主播在线| 精品高潮呻吟久久av| 精品国产高潮中文字幕| 日韩中文字幕精品淫| 人妻av无码专区久久绿巨人| 国产使劲操在线播放| 成人亚洲国产综合精品| 亚洲精品久久综合久| 人妻最新视频在线免费观看| eeuss鲁片一区二区三区| 神马午夜在线观看视频| 亚洲国产欧美一区二区三区久久| 天天射夜夜操狠狠干| 中文字幕高清在线免费播放| 久久久久久久99精品| 日本一二三中文字幕| 日韩人妻xxxxx| 涩涩的视频在线观看视频| 日本女人一级免费片| 亚洲 人妻 激情 中文| 中文字幕国产专区欧美激情| 久久热这里这里只有精品| 亚洲国产在线精品国偷产拍| 91国产资源在线视频| 女生被男生插的视频网站| 欧美成一区二区三区四区| 国产欧美日韩在线观看不卡| 天码人妻一区二区三区在线看| 精品久久久久久久久久中文蒉| 风流唐伯虎电视剧在线观看 | 亚洲伊人久久精品影院一美女洗澡| 在线观看911精品国产| 精品av久久久久久久| 青娱乐蜜桃臀av色| 天美传媒mv视频在线观看| 在线国产日韩欧美视频| 亚洲中文字字幕乱码| 亚洲精品午夜aaa久久| 97人人模人人爽人人喊| 国产美女精品福利在线| 亚洲在线一区二区欧美| 日韩美女精品视频在线观看网站| 天天做天天干天天舔| 亚洲国产成人av在线一区| 姐姐的朋友2在线观看中文字幕 | av在线免费资源站| 免费av岛国天堂网站| 亚洲天天干 夜夜操| 精品suv一区二区69| 国产又粗又硬又猛的毛片视频| 经典亚洲伊人第一页| 一级黄色av在线观看| 亚洲嫩模一区二区三区| 天干天天天色天天日天天射 | 初美沙希中文字幕在线| 午夜av一区二区三区| 把腿张开让我插进去视频| 色爱av一区二区三区| 天堂av狠狠操蜜桃| 在线观看一区二区三级| 国产麻豆国语对白露脸剧情| 精品av久久久久久久| 黑人乱偷人妻中文字幕| 欧美viboss性丰满| 国产麻豆91在线视频| 777奇米久久精品一区| 亚洲高清国产一区二区三区| 国产精彩对白一区二区三区| 亚洲护士一区二区三区| 99久久中文字幕一本人| 97超碰免费在线视频| 91免费黄片可看视频| 韩国黄色一级二级三级| 自拍偷拍日韩欧美亚洲| 色噜噜噜噜18禁止观看| 啊啊啊想要被插进去视频| 少妇高潮一区二区三区| 熟女在线视频一区二区三区| 97人人模人人爽人人喊| 和邻居少妇愉情中文字幕| 一区二区三区美女毛片| yellow在线播放av啊啊啊| 欧美日韩在线精品一区二区三| 孕妇奶水仑乱A级毛片免费看| 黄色大片免费观看网站| 成年人啪啪视频在线观看| 亚洲蜜臀av一区二区三区九色| 亚洲一级特黄特黄黄色录像片| 888欧美视频在线| 激情人妻校园春色亚洲欧美 | 黑人巨大的吊bdsm| 亚洲中文字幕综合小综合| 亚洲欧洲一区二区在线观看| 亚洲成人av一区久久| 社区自拍揄拍尻屁你懂的| 狠狠的往里顶撞h百合| 美女 午夜 在线视频| 93精品视频在线观看| 日韩成人性色生活片| 直接能看的国产av| 适合午夜一个人看的视频| 绝色少妇高潮3在线观看| 亚洲人妻国产精品综合| 偷拍3456eee| 亚洲国产成人在线一区| 色在线观看视频免费的| 亚洲的电影一区二区三区| 天天日天天干天天要| 伊人开心婷婷国产av| 亚洲免费福利一区二区三区| 大黑人性xxxxbbbb| 亚洲少妇高潮免费观看| 3D动漫精品啪啪一区二区下载| 日本韩国免费一区二区三区视频| 亚洲av自拍偷拍综合| 日韩美在线观看视频黄| 午夜精品久久久久久99热| 国产三级影院在线观看| 不卡一区一区三区在线| 国产成人综合一区2区| 精品一区二区亚洲欧美| 99热久久这里只有精品8| 骚货自慰被发现爆操| 日美女屁股黄邑视频| 精品久久久久久高潮| 国产成人无码精品久久久电影 | 亚洲成人免费看电影| 偷拍自拍亚洲美腿丝袜| 青青尤物在线观看视频网站| 亚洲免费视频欧洲免费视频| 欧美精品亚洲精品日韩在线| 亚洲精品乱码久久久本| 国产熟妇一区二区三区av| 日韩欧美国产一区ab| 午夜国产福利在线观看| 天天躁夜夜躁日日躁a麻豆| 亚洲国产精品久久久久蜜桃| 99一区二区在线观看| 精品久久婷婷免费视频| 自拍 日韩 欧美激情| 98精产国品一二三产区区别| 亚洲精品ww久久久久久| 午夜在线观看一区视频| 夜夜嗨av蜜臀av| 国内自拍第一页在线观看| 亚洲av黄色在线网站| 久久久久久久精品成人热| 午夜dv内射一区区| tube69日本少妇| 顶级尤物粉嫩小尤物网站| 水蜜桃国产一区二区三区| 亚洲精品乱码久久久本| av手机在线免费观看日韩av| 宅男噜噜噜666免费观看| 自拍偷拍,中文字幕| 免费看高清av的网站| 夜夜骑夜夜操夜夜奸| 久久三久久三久久三久久| 揄拍成人国产精品免费看视频| 亚洲欧美日韩视频免费观看| 国产真实灌醉下药美女av福利| 日韩欧美亚洲熟女人妻| 中文字幕人妻av在线观看| 精品人妻一二三区久久| 中文字幕无码日韩专区免费| 在线视频精品你懂的| 九一传媒制片厂视频在线免费观看| 日本高清在线不卡一区二区| 91啪国自产中文字幕在线| 蜜桃久久久久久久人妻| 大尺度激情四射网站| av手机在线观播放网站| 国产性生活中老年人视频网站| 日本丰满熟妇BBXBBXHD| 91精品免费久久久久久| 自拍偷拍亚洲另类色图| 在线播放 日韩 av| 最新欧美一二三视频| 免费费一级特黄真人片| 亚洲va国产va欧美va在线| 一区二区三区的久久的蜜桃的视频| 不卡一区一区三区在线| 亚洲综合乱码一区二区| 亚洲国产精品美女在线观看| 国产V亚洲V天堂无码欠欠| 国产麻豆精品人妻av| 国产精品一二三不卡带免费视频| 91中文字幕最新合集| 国产精品精品精品999| 在线观看av观看av| 中文字幕av第1页中文字幕| 一区二区三区久久久91| 91大屁股国产一区二区| 1区2区3区不卡视频| 国产麻豆精品人妻av| 欧美日本国产自视大全| 四川乱子伦视频国产vip| 在线免费观看靠比视频的网站| 91天堂精品一区二区| 啪啪啪啪啪啪啪啪啪啪黄色| 久草福利电影在线观看| 精品一区二区三区三区色爱| 亚洲av日韩高清hd| 中文字幕日韩精品就在这里| 天堂av狠狠操蜜桃| 亚洲午夜精品小视频| 中文字幕日韩无敌亚洲精品| 88成人免费av网站| 国产精品伦理片一区二区| 97精品成人一区二区三区 | 免费男阳茎伸入女阳道视频| 国产三级片久久久久久久| 欧美视频中文一区二区三区| 93视频一区二区三区| 国产成人精品午夜福利训2021| 色偷偷伊人大杳蕉综合网| 婷婷五月亚洲综合在线| 黄色的网站在线免费看| 亚洲图片欧美校园春色| 国产黄色大片在线免费播放| 成年人中文字幕在线观看| 亚洲av成人网在线观看| 插逼视频双插洞国产操逼插洞| 日韩精品中文字幕播放| 午夜精品亚洲精品五月色| 综合精品久久久久97| 欧美亚洲一二三区蜜臀| 91久久人澡人人添人人爽乱| 日韩美女精品视频在线观看网站| 国产成人精品亚洲男人的天堂| 国产九色91在线观看精品| 日本性感美女三级视频| 99久久成人日韩欧美精品| 天天干天天操天天插天天日| 中字幕人妻熟女人妻a62v网 | 国产精品国产三级国产精东 | 六月婷婷激情一区二区三区| 五月天色婷婷在线观看视频免费| 骚货自慰被发现爆操| 日韩美女福利视频网| 欧美日韩高清午夜蜜桃大香蕉| 中国视频一区二区三区| 熟女人妻三十路四十路人妻斩| 欧美综合婷婷欧美综合| 国产精品黄大片在线播放| 美女福利写真在线观看视频| 国产熟妇一区二区三区av| 韩国女主播精品视频网站| 成人性爱在线看四区| 肏插流水妹子在线乐播下载| 精彩视频99免费在线| 97欧洲一区二区精品免费| 黑人进入丰满少妇视频| 亚洲色偷偷综合亚洲AV伊人| 日本av在线一区二区三区| 天天日天天天天天天天天天天 | 亚洲 欧美 精品 激情 偷拍| 99re国产在线精品| 日日夜夜大香蕉伊人| 国产亚洲精品欧洲在线观看| 91综合久久亚洲综合| 欧美精品伦理三区四区| 久久久久久9999久久久久| 偷拍美女一区二区三区| 亚洲成人午夜电影在线观看| 99精品亚洲av无码国产另类| 91极品大一女神正在播放| 超级福利视频在线观看| 成年人该看的视频黄免费| 夜夜嗨av一区二区三区中文字幕| 日本少妇精品免费视频| 成人av在线资源网站| 亚洲天堂精品久久久| 午夜久久香蕉电影网| 久久这里有免费精品| 少妇高潮无套内谢麻豆| 欧美成人综合色在线噜噜| 免费在线福利小视频| 精品人妻伦一二三区久| 国产视频在线视频播放| 日韩北条麻妃一区在线| 久久美欧人妻少妇一区二区三区| 中文字幕亚洲久久久| 一区二区三区四区五区性感视频| 中英文字幕av一区| 91精品视频在线观看免费| 中文字幕一区二区三区蜜月| 午夜dv内射一区区| 日本熟妇丰满厨房55| 久久久极品久久蜜桃| 中文字幕,亚洲人妻| 99久久99一区二区三区| 国产美女一区在线观看| 中文字幕视频一区二区在线观看| 在线网站你懂得老司机| 欧美一区二区中文字幕电影| 欧洲亚洲欧美日韩综合| 国内资源最丰富的网站| 国产亚洲精品品视频在线| 亚洲一区二区久久久人妻| 亚洲综合一区成人在线| 狠狠躁狠狠爱网站视频| 1000小视频在线| 涩涩的视频在线观看视频| 亚洲另类在线免费观看| av手机免费在线观看高潮| 日韩一区二区电国产精品| 精品suv一区二区69| 日韩国产乱码中文字幕| 亚洲男人让女人爽的视频| 亚洲嫩模一区二区三区| 男人的网址你懂的亚洲欧洲av| 阴茎插到阴道里面的视频| aiss午夜免费视频| 亚洲人成精品久久久久久久| 久久综合老鸭窝色综合久久| 亚洲 欧美 精品 激情 偷拍 | 特一级特级黄色网片| 青青尤物在线观看视频网站| 亚洲欧美另类自拍偷拍色图| 亚洲国产欧美一区二区丝袜黑人| 性欧美日本大妈母与子| 成年人黄视频在线观看| 亚洲特黄aaaa片| 操操网操操伊剧情片中文字幕网| jiujiure精品视频在线| 不卡一不卡二不卡三| 中文字幕一区二 区二三区四区| 中国产一级黄片免费视频播放| 亚洲一区二区三区在线高清| 伊人精品福利综合导航| 婷婷激情四射在线观看视频| av天堂中文字幕最新| 天天干狠狠干天天操| 啊慢点鸡巴太大了啊舒服视频| 天天操天天弄天天射| 国产熟妇人妻ⅹxxxx麻豆| 中字幕人妻熟女人妻a62v网 | 国产亚洲成人免费在线观看| 夜色撩人久久7777| 国产又粗又硬又猛的毛片视频| 午夜国产免费福利av| 国产日韩欧美视频在线导航| 亚洲视频在线视频看视频在线| 人人超碰国字幕观看97| 2022天天干天天操| 国产午夜亚洲精品麻豆| 亚洲成人av在线一区二区| 青青草在观免费国产精品| 国产在线拍揄自揄视频网站| 精品国产在线手机在线| 91av精品视频在线| av欧美网站在线观看| 美日韩在线视频免费看| 漂亮 人妻被中出中文| 亚洲av无硬久久精品蜜桃| 丝袜肉丝一区二区三区四区在线 | 大香蕉伊人中文字幕| 又粗又硬又猛又黄免费30| 在线播放国产黄色av| 国产精品大陆在线2019不卡| 欧美性感尤物人妻在线免费看| 亚洲精品av在线观看| 亚洲人妻视频在线网| 亚洲视频在线观看高清| 青青青青青青青青青青草青青| 免费看高清av的网站| 精品久久久久久高潮| 色哟哟在线网站入口| 国产精品一区二区久久久av| 蜜桃视频在线欧美一区| 女生自摸在线观看一区二区三区 | 快点插进来操我逼啊视频| 午夜精品一区二区三区4| 青青社区2国产视频| 亚洲色偷偷综合亚洲AV伊人| 5528327男人天堂| 欧美va不卡视频在线观看| 欧美老鸡巴日小嫩逼| 国产日本欧美亚洲精品视| 91香蕉成人app下载| 亚洲精品麻豆免费在线观看| 黄色av网站免费在线| 国产丰满熟女成人视频| 亚洲高清自偷揄拍自拍| 淫秽激情视频免费观看| 日本xx片在线观看| 18禁精品网站久久| 日本午夜福利免费视频| 一区二区熟女人妻视频| 一本久久精品一区二区| 欧美精品久久久久久影院| 人妻丝袜榨强中文字幕| 人妻熟女在线一区二区| 日韩精品二区一区久久| 好吊操视频这里只有精品| 欧美精品中文字幕久久二区| 特级欧美插插插插插bbbbb| www天堂在线久久| 国产日韩一区二区在线看| 班长撕开乳罩揉我胸好爽| 国产熟妇人妻ⅹxxxx麻豆| 乱亲女秽乱长久久久| 福利在线视频网址导航| 亚洲精品乱码久久久本| yellow在线播放av啊啊啊| 福利视频一区二区三区筱慧| 免费在线播放a级片| 亚洲欧美激情国产综合久久久| 麻豆精品成人免费视频| 91一区精品在线观看| 久草视频在线看免费| 女警官打开双腿沦为性奴| 老司机在线精品福利视频| 人妻无码中文字幕专区| 欧美日韩在线精品一区二区三| 自拍偷拍vs一区二区三区| 日韩欧美在线观看不卡一区二区| 青青草人人妻人人妻| 自拍偷拍日韩欧美一区二区| 偷拍自拍亚洲视频在线观看| 青青伊人一精品视频| 欧美色呦呦最新网址| 天天摸天天干天天操科普| 青青草亚洲国产精品视频| 日本www中文字幕| 成年人黄色片免费网站| 日韩人妻在线视频免费| 91九色porny国产蝌蚪视频| 亚洲av日韩精品久久久| 国产高清97在线观看视频| 中文字幕乱码人妻电影| 国产午夜亚洲精品不卡在线观看| 日本女大学生的黄色小视频| 日韩一个色综合导航| 狍和女人的王色毛片| 沙月文乃人妻侵犯中文字幕在线 | 精品久久久久久久久久久久人妻| 中文字幕 人妻精品| 亚洲一级美女啪啪啪| 中文字幕在线第一页成人| 丰满的子国产在线观看| 亚洲国产在线精品国偷产拍| 亚洲成人线上免费视频观看| 粉嫩av蜜乳av蜜臀| 亚洲精品无码久久久久不卡| 污污小视频91在线观看| 精品人妻一二三区久久| 中文字幕熟女人妻久久久| 亚洲最大黄 嗯色 操 啊| 国产女人露脸高潮对白视频| 精品一区二区三区三区色爱| 免费在线观看污污视频网站| 天美传媒mv视频在线观看| 在线观看成人国产电影| 少妇与子乱在线观看| 欧美一区二区三区乱码在线播放 | 久久热这里这里只有精品| 大肉大捧一进一出好爽在线视频| 扒开让我视频在线观看| 四川乱子伦视频国产vip| 老司机午夜精品视频资源 | 视频一区二区综合精品| 亚洲视频乱码在线观看| 99精品国产aⅴ在线观看| 日本午夜爽爽爽爽爽视频在线观看 | 99热久久极品热亚洲| 在线观看的a站 最新| 天天干天天操天天插天天日| 天天插天天狠天天操| 亚洲高清免费在线观看视频| 免费大片在线观看视频网站| 青青色国产视频在线| 女同久久精品秋霞网| 中文字幕无码日韩专区免费| 日韩精品中文字幕在线| 中文字幕日本人妻中出| 亚洲综合一区成人在线| 中文字幕+中文字幕| 午夜精品久久久久久99热| 后入美女人妻高清在线| av在线资源中文字幕| 东游记中文字幕版哪里可以看到 | 加勒比视频在线免费观看| 中文字幕日韩人妻在线三区| 国产日韩精品电影7777| 欧美成人一二三在线网| 岛国av高清在线成人在线| 老司机午夜精品视频资源| 天天干天天操天天玩天天射 | 美女张开两腿让男人桶av| 成人H精品动漫在线无码播放| ka0ri在线视频| 蜜臀av久久久久蜜臀av麻豆| 青青青青爽手机在线| 不戴胸罩引我诱的隔壁的人妻| 91国内视频在线观看| 亚洲欧美另类自拍偷拍色图| 欧美精品亚洲精品日韩在线| 欧美国品一二三产区区别| 婷婷五月亚洲综合在线| 亚洲护士一区二区三区| 久久久久久久久久一区二区三区 | 人人在线视频一区二区| 中文字幕在线永久免费播放| 换爱交换乱高清大片| 欧美viboss性丰满| sw137 中文字幕 在线| rct470中文字幕在线| 亚洲av日韩精品久久久| 久久丁香婷婷六月天| 亚洲高清国产自产av| 国产亚洲视频在线二区| 1区2区3区4区视频在线观看| 国产精品探花熟女在线观看| tube69日本少妇| 最新的中文字幕 亚洲| 操的小逼流水的文章| 人妻自拍视频中国大陆| 人人人妻人人澡人人| 操日韩美女视频在线免费看| 中文字幕之无码色多多| 黑人性生活视频免费看| 黄工厂精品视频在线观看| 欧美日韩人妻久久精品高清国产| 亚洲欧美色一区二区| 国产av自拍偷拍盛宴| 插逼视频双插洞国产操逼插洞| 在线观看黄色成年人网站| 老司机99精品视频在线观看| 久草视频福利在线首页| 中文字幕午夜免费福利视频| 动漫av网站18禁| 被大鸡吧操的好舒服视频免费| 精品高跟鞋丝袜一区二区| 一区二区三区毛片国产一区| 日韩影片一区二区三区不卡免费| 91chinese在线视频| 欧美日本在线视频一区| 国产日本欧美亚洲精品视| 午夜dv内射一区区| 91九色porny国产在线| 搞黄色在线免费观看| 欧洲欧美日韩国产在线| 国产视频网站一区二区三区| 欧美viboss性丰满| 久久机热/这里只有| 久久久人妻一区二区| 成年人黄色片免费网站| 熟女少妇激情五十路| 久久久久久9999久久久久| 中文字幕人妻av在线观看| 中文人妻AV久久人妻水| 亚洲av黄色在线网站| www天堂在线久久| 无套猛戳丰满少妇人妻| 欧美黑人性猛交xxxxⅹooo| 日本午夜爽爽爽爽爽视频在线观看| 狠狠嗨日韩综合久久| 亚洲另类伦春色综合小| 亚洲天堂成人在线观看视频网站| 亚洲人妻30pwc| 93精品视频在线观看| 午夜精品一区二区三区城中村| 欧美色婷婷综合在线| 91大屁股国产一区二区| 操人妻嗷嗷叫视频一区二区| 国产精品黄片免费在线观看| av一区二区三区人妻| 人妻少妇中文有码精品| 77久久久久国产精产品| 自拍偷拍日韩欧美亚洲| 国产一区二区久久久裸臀| 欧美精品免费aaaaaa| 人妻激情图片视频小说| 大胸性感美女羞爽操逼毛片| 黄色视频在线观看高清无码| 国产综合精品久久久久蜜臀| 日韩a级精品一区二区| 国产又粗又硬又猛的毛片视频| 久久丁香婷婷六月天| 88成人免费av网站| av在线shipin| 在线免费观看欧美小视频| 国产一级精品综合av| 国产女人被做到高潮免费视频 | 欧美专区第八页一区在线播放| 国产又色又刺激在线视频| 蜜桃视频在线欧美一区| 成人H精品动漫在线无码播放| 在线视频国产欧美日韩| 欧美色婷婷综合在线| www日韩毛片av| 国产在线免费观看成人| 国内自拍第一页在线观看| 日本一二三区不卡无| 成人蜜桃美臀九一一区二区三区 | eeuss鲁片一区二区三区| 国产精品人妻一区二区三区网站 | 人妻熟女中文字幕aⅴ在线| 中国老熟女偷拍第一页| 性色蜜臀av一区二区三区| 成人资源在线观看免费官网| 极品丝袜一区二区三区| 夜女神免费福利视频| 天天干天天操天天插天天日| 超pen在线观看视频公开97| 天天日天天玩天天摸| av男人天堂狠狠干| 97年大学生大白天操逼| 97国产在线av精品| 狠狠躁夜夜躁人人爽天天久天啪| 狠狠躁狠狠爱网站视频| weyvv5国产成人精品的视频| 高潮喷水在线视频观看| 中文字幕网站你懂的| 天天日天天爽天天干| 中文字幕 人妻精品| 日韩一个色综合导航| 亚洲天堂第一页中文字幕| 人妻素人精油按摩中出| 亚洲成av人无码不卡影片一| 精品日产卡一卡二卡国色天香| 亚洲 欧美 自拍 偷拍 在线| 在线观看亚洲人成免费网址| 91麻豆精品久久久久| 国产午夜亚洲精品麻豆| 亚洲高清自偷揄拍自拍| 天天操天天干天天艹| 蜜臀av久久久久久久| 少妇高潮一区二区三区| 九色视频在线观看免费| 人人爱人人妻人人澡39| 欧美伊人久久大香线蕉综合| 国产真实灌醉下药美女av福利| 亚洲无码一区在线影院| 午夜婷婷在线观看视频| 国产精品黄页网站视频| avjpm亚洲伊人久久| 亚洲熟女女同志女同| 中文字幕人妻熟女在线电影| 伊人开心婷婷国产av| 亚洲第一伊人天堂网| 免费在线播放a级片| 亚洲av极品精品在线观看| 青青青青青青青在线播放视频| 精品欧美一区二区vr在线观看| 亚欧在线视频你懂的| 日韩国产乱码中文字幕| 亚洲成人激情视频免费观看了| 亚洲无码一区在线影院| av手机在线观播放网站| 亚洲人人妻一区二区三区| 欧美综合婷婷欧美综合| 91精品国产麻豆国产| 自拍偷区二区三区麻豆| 日本少妇人妻xxxxxhd| 久久久精品999精品日本 | 国产1区,2区,3区| 成人30分钟免费视频| 亚洲av可乐操首页| 只有精品亚洲视频在线观看| 在线新三级黄伊人网| 欧美在线精品一区二区三区视频| 国产成人午夜精品福利| 97国产在线av精品| 偷拍自拍视频图片免费| 宅男噜噜噜666国产| 最近的中文字幕在线mv视频| 亚洲国产精品美女在线观看| 91人妻人人做人人爽在线| 欧美精品伦理三区四区| 国产黄网站在线观看播放| 国产片免费观看在线观看| 国产乱弄免费视频观看| 国产精彩福利精品视频| 亚洲精品三级av在线免费观看| 又大又湿又爽又紧A视频| av天堂中文免费在线| 啪啪啪啪啪啪啪免费视频| 阿v天堂2014 一区亚洲| 98精产国品一二三产区区别| 久久精品国产999| 美女福利视频导航网站 | 黑人借宿ntr人妻的沦陷2| 91香蕉成人app下载| 久久机热/这里只有| 狠狠躁狠狠爱网站视频| 日韩av大胆在线观看| 人妻少妇亚洲精品中文字幕| 午夜精品福利一区二区三区p | 青青青青青免费视频| 欧洲精品第一页欧洲精品亚洲| 91国产在线免费播放| 欧美国产亚洲中英文字幕| 国产激情av网站在线观看| 中国黄色av一级片| 五十路丰满人妻熟妇| 亚洲综合图片20p| 国产女孩喷水在线观看| 欧美在线精品一区二区三区视频| 亚洲免费在线视频网站| 啊啊好大好爽啊啊操我啊啊视频 | 521精品视频在线观看| 亚洲一区二区三区精品视频在线| 黄色资源视频网站日韩| 天天干狠狠干天天操| 适合午夜一个人看的视频| 日韩欧美在线观看不卡一区二区| 80电影天堂网官网| 欧美xxx成人在线| 一级A一级a爰片免费免会员| 亚洲特黄aaaa片| 特大黑人巨大xxxx| av老司机亚洲一区二区| 国产va精品免费观看| 操的小逼流水的文章| 999久久久久999| 东京干手机福利视频| 人妻凌辱欧美丰满熟妇| 国产之丝袜脚在线一区二区三区| 色婷婷久久久久swag精品| 欧美亚洲自偷自拍 在线| 欧美日韩熟女一区二区三区| 久久久久久国产精品| 国产精品系列在线观看一区二区| 热思思国产99re| 日本一道二三区视频久久| 日本女大学生的黄色小视频| 999热精品视频在线| 亚洲熟色妇av日韩熟色妇在线 | chinese国产盗摄一区二区| 唐人色亚洲av嫩草| 人妻少妇av在线观看| 天天干天天搞天天摸| 国产亚洲精品欧洲在线观看| 3337p日本欧洲大胆色噜噜| 天天日天天玩天天摸| 天天日天天玩天天摸| 大香蕉日本伊人中文在线| www日韩毛片av| 欧美精品亚洲精品日韩在线| 韩国AV无码不卡在线播放| 中文字幕乱码av资源| 三级等保密码要求条款| 国产成人精品久久二区91| 国产在线自在拍91国语自产精品| av久久精品北条麻妃av观看| 精品国产在线手机在线| 日韩成人性色生活片| 大胆亚洲av日韩av| 午夜美女福利小视频| www,久久久,com| 亚洲成人国产av在线| 特一级特级黄色网片| 麻豆性色视频在线观看| 人妻自拍视频中国大陆| 亚洲综合自拍视频一区| 久碰精品少妇中文字幕av| 大黑人性xxxxbbbb| 一级黄色片夫妻性生活| 亚洲综合在线观看免费| 一区二区久久成人网| 成人av久久精品一区二区| 国产高清女主播在线| 亚洲欧美综合在线探花| 午夜成午夜成年片在线观看| 亚洲激情偷拍一区二区| av高潮迭起在线观看| 天天想要天天操天天干| 日本av高清免费网站| 在线免费观看av日韩| 日本丰满熟妇大屁股久久| 大学生A级毛片免费视频| 国产第一美女一区二区三区四区| 中国老熟女偷拍第一页| 欧美黑人性暴力猛交喷水| 国产伊人免费在线播放| 福利午夜视频在线合集| 少妇一区二区三区久久久| 国产福利小视频二区| 国产黄色高清资源在线免费观看| 自拍偷拍 国产资源| 亚洲 中文字幕在线 日韩| 91人妻精品久久久久久久网站| 快插进小逼里大鸡吧视频| 欧美精品欧美极品欧美视频| 97a片免费在线观看| 麻豆精品成人免费视频| 大屁股熟女一区二区三区| 涩爱综合久久五月蜜臀| 少妇ww搡性bbb91| 北条麻妃av在线免费观看| 日本三极片中文字幕| 九九热99视频在线观看97| 免费人成黄页网站在线观看国产| 久久综合老鸭窝色综合久久| 91香蕉成人app下载| 国产亚洲欧美另类在线观看| 少妇人妻100系列| 免费在线播放a级片| 色婷婷久久久久swag精品| 青青青青爽手机在线| 久久久久久九九99精品| 欧美视频综合第一页| 2020久久躁狠狠躁夜夜躁| 男生用鸡操女生视频动漫 | av高潮迭起在线观看| 非洲黑人一级特黄片| 美女在线观看日本亚洲一区| 可以在线观看的av中文字幕| 日本美女性生活一级片| 一区二区视频视频视频| 国产男女视频在线播放| 日本少妇人妻xxxxx18| 一区二区三区在线视频福利| 精品一区二区亚洲欧美| 动漫av网站18禁| 99精品亚洲av无码国产另类| 成人av久久精品一区二区| 天天日天天操天天摸天天舔| 91 亚洲视频在线观看| 欧美亚洲自偷自拍 在线| 91久久综合男人天堂| 最后99天全集在线观看| 一区二区三区美女毛片| brazzers欧熟精品系列| 成人激情文学网人妻| 成年人黄色片免费网站| 青青青国产免费视频| 爱有来生高清在线中文字幕| 中文字幕亚洲久久久| 在线新三级黄伊人网| 中文字幕熟女人妻久久久| 97成人免费在线观看网站| 超pen在线观看视频公开97| 三上悠亚和黑人665番号| 丝袜亚洲另类欧美变态| 美女福利视频导航网站| 社区自拍揄拍尻屁你懂的| 热久久只有这里有精品| 人妻无码色噜噜狠狠狠狠色| 欧美综合婷婷欧美综合| 欧洲欧美日韩国产在线| 日日夜夜狠狠干视频| 天天日天天透天天操| tube69日本少妇| 青青青视频手机在线观看| 搞黄色在线免费观看| 亚洲欧美另类自拍偷拍色图| 2020av天堂网在线观看| 亚洲最大免费在线观看| 亚洲精品国产综合久久久久久久久| 黄色片黄色片wyaa| 成人激情文学网人妻| 天堂av在线最新版在线| 成年人黄色片免费网站| 国产美女午夜福利久久| 在线新三级黄伊人网| 亚洲第一伊人天堂网| 香蕉片在线观看av| 成人福利视频免费在线| 在线观看免费视频网| 天天日天天操天天摸天天舔| 亚洲av黄色在线网站| 久久香蕉国产免费天天| 91久久精品色伊人6882| 搞黄色在线免费观看| 精品成人午夜免费看| 99精品国产aⅴ在线观看| 9l人妻人人爽人人爽| 黑人大几巴狂插日本少妇| 超碰中文字幕免费观看| 亚洲精品久久视频婷婷| 欧美xxx成人在线| 91国内视频在线观看| 亚洲熟女女同志女同| 中文字母永久播放1区2区3区| 在线免费视频 自拍| 综合激情网激情五月五月婷婷| 2022天天干天天操| 青青青激情在线观看视频| 中文字幕日韩无敌亚洲精品| 美女吃鸡巴操逼高潮视频| 真实国模和老外性视频| 大香蕉大香蕉大香蕉大香蕉大香蕉| 青青青激情在线观看视频| 亚洲一区二区三区五区| 制服丝袜在线人妻中文字幕| 天天日天天日天天擦| 中文字幕奴隷色的舞台50| 91社福利《在线观看| www久久久久久久久久久| 99亚洲美女一区二区三区| 亚洲在线一区二区欧美| 老司机99精品视频在线观看| 九色porny九色9l自拍视频| 绝色少妇高潮3在线观看| aⅴ精产国品一二三产品| 中文字幕一区二区三区人妻大片| 岛国青草视频在线观看| 国产亚洲成人免费在线观看 | 成人av久久精品一区二区| 日韩精品二区一区久久| 国产97视频在线精品| 91成人精品亚洲国产| 性生活第二下硬不起来| 丰满少妇翘臀后进式| 亚洲少妇高潮免费观看| 亚洲熟妇无码一区二区三区| 日本丰满熟妇BBXBBXHD| 日本精品一区二区三区在线视频。 | 日日夜夜大香蕉伊人| 亚洲青青操骚货在线视频| 精品美女久久久久久| 无套猛戳丰满少妇人妻| 精品黑人一区二区三区久久国产| 午夜频道成人在线91| 懂色av蜜桃a v| 亚洲va国产va欧美精品88| 夜女神免费福利视频| 婷婷色中文亚洲网68| 桃色视频在线观看一区二区 | 亚洲护士一区二区三区| 亚洲高清国产拍青青草原| 欧美日韩一级黄片免费观看| 丝袜肉丝一区二区三区四区在线| 无忧传媒在线观看视频| 3344免费偷拍视频| gay gay男男瑟瑟在线网站| 婷婷六月天中文字幕| aⅴ精产国品一二三产品| 9色在线视频免费观看| 中文字幕在线欧美精品| 亚洲国产精品中文字幕网站| 韩国女主播精品视频网站| av在线免费观看亚洲天堂| 日韩无码国产精品强奸乱伦| 哥哥姐姐综合激情小说| 巨乳人妻日下部加奈被邻居中出 | 亚洲av日韩av网站| 日韩欧美高清免费在线| 超碰在线中文字幕一区二区| 18禁美女羞羞免费网站| 天天摸天天亲天天舔天天操天天爽 | 狍和女人的王色毛片| 在线国产日韩欧美视频| 亚洲精品国产久久久久久| 国产第一美女一区二区三区四区| 午夜福利资源综合激情午夜福利资| 女同久久精品秋霞网| 亚洲护士一区二区三区| 99热色原网这里只有精品| 蜜臀av久久久久久久| 51国产偷自视频在线播放| 日韩三级电影华丽的外出| 国产欧美精品免费观看视频| 中文字幕亚洲久久久| 骚逼被大屌狂草视频免费看| 四川乱子伦视频国产vip| 美女福利视频网址导航| 日本人妻少妇18—xx| 粉嫩小穴流水视频在线观看| 特级欧美插插插插插bbbbb| 瑟瑟视频在线观看免费视频| 亚洲 清纯 国产com| 日韩精品中文字幕在线| 看一级特黄a大片日本片黑人| 六月婷婷激情一区二区三区| AV无码一区二区三区不卡| 日韩av中文在线免费观看| 鸡巴操逼一级黄色气| 岛国毛片视频免费在线观看| 人人人妻人人澡人人| 亚洲av极品精品在线观看| 日韩精品二区一区久久| 小穴多水久久精品免费看| 国产成人精品一区在线观看| 99热久久这里只有精品8| 亚洲欧美激情人妻偷拍| 国产熟妇人妻ⅹxxxx麻豆| 视频在线亚洲一区二区| 人妻自拍视频中国大陆| 天天日天天透天天操| 区一区二区三国产中文字幕| 日韩欧美一级aa大片| 久久丁香婷婷六月天| 亚洲偷自拍高清视频| lutube在线成人免费看| 蝴蝶伊人久久中文娱乐网| 精品区一区二区三区四区人妻| 天天日天天日天天擦| 亚洲av第国产精品| 国产黄网站在线观看播放| 国产精品自偷自拍啪啪啪| 99精品国产免费久久| 国产精品成久久久久三级蜜臀av| 美女张开腿让男生操在线看| 黄色片年轻人在线观看| 懂色av之国产精品| 日韩亚洲高清在线观看| 亚洲国产最大av综合| 中文字幕1卡1区2区3区| 初美沙希中文字幕在线| 色吉吉影音天天干天天操| 亚洲人一区二区中文字幕| 美女被肏内射视频网站| 国产品国产三级国产普通话三级| 2018最新中文字幕在线观看| 成人av中文字幕一区| 久碰精品少妇中文字幕av | 亚洲精品麻豆免费在线观看| 免费在线观看污污视频网站| 亚洲国产免费av一区二区三区| 亚洲成人黄色一区二区三区| 性色蜜臀av一区二区三区| 国产在线免费观看成人| 2022国产综合在线干| 国产自拍黄片在线观看| 人妻少妇精品久久久久久| 精品人人人妻人人玩日产欧| 亚洲成人av在线一区二区| 国产不卡av在线免费| 久久综合老鸭窝色综合久久| 青青操免费日综合视频观看| 国产成人午夜精品福利| 午夜精品久久久久久99热| 亚洲一区二区三区uij| 亚洲精品国产在线电影| 人人在线视频一区二区| 国产中文字幕四区在线观看| 天天日天天玩天天摸| 青娱乐蜜桃臀av色| 色婷婷久久久久swag精品| 福利视频广场一区二区| 蜜桃视频17c在线一区二区| 一区二区三区另类在线| 天天色天天操天天透| 午夜在线观看岛国av,com| 日本午夜福利免费视频| 99精品国产自在现线观看| 在线观看成人国产电影| 一本一本久久a久久精品综合不卡| 硬鸡巴动态操女人逼视频| 狠狠躁狠狠爱网站视频| 51国产成人精品视频| 大屁股肉感人妻中文字幕在线| 不卡一不卡二不卡三| 一个色综合男人天堂| avjpm亚洲伊人久久| 黑人变态深video特大巨大| 中字幕人妻熟女人妻a62v网 | 午夜在线精品偷拍一区二| 懂色av之国产精品| 80电影天堂网官网| 一区二区三区的久久的蜜桃的视频| 2021年国产精品自拍| 男人的天堂av日韩亚洲| 青青青艹视频在线观看| 天天夜天天日天天日| japanese日本熟妇另类| 视频一区 视频二区 视频| 亚洲免费av在线视频| 手机看片福利盒子日韩在线播放| 非洲黑人一级特黄片| 中文字幕av熟女人妻| 日本午夜久久女同精女女| 91免费观看国产免费| 亚洲欧美人精品高清| 日韩欧美高清免费在线 | 岛国免费大片在线观看 | 欧美激情精品在线观看| 亚洲一区久久免费视频| 特级欧美插插插插插bbbbb| 成人av电影免费版| 亚洲综合在线观看免费| 91免费观看国产免费| 91综合久久亚洲综合| 一区二区在线视频中文字幕| 天天干天天日天天干天天操| 青青草视频手机免费在线观看| 快插进小逼里大鸡吧视频| 区一区二区三国产中文字幕| 天天摸天天日天天操| 特级无码毛片免费视频播放| 888欧美视频在线| 国产av一区2区3区| 女同互舔一区二区三区| 经典亚洲伊人第一页| 成人综合亚洲欧美一区| 成人30分钟免费视频| 97青青青手机在线视频| 老司机99精品视频在线观看| 国产品国产三级国产普通话三级| 人妻少妇中文有码精品| 日韩av熟妇在线观看| 亚洲中文字幕国产日韩| 国产男女视频在线播放| 9国产精品久久久久老师| 久久永久免费精品人妻专区 | 欧美精品免费aaaaaa| 亚洲国产精品久久久久久6| 1区2区3区不卡视频| 欧美视频综合第一页| 欧美一区二区中文字幕电影 | 视频一区 二区 三区 综合| 98视频精品在线观看| 亚洲自拍偷拍综合色| 后入美女人妻高清在线| 国产精品一区二区av国| 中文字幕av男人天堂| 三上悠亚和黑人665番号| 国产普通话插插视频| 亚洲另类图片蜜臀av| 精品人妻一二三区久久| 热久久只有这里有精品| 国产麻豆91在线视频| 亚洲偷自拍高清视频| 国产亚洲欧美另类在线观看| yy96视频在线观看| 亚洲推理片免费看网站| 日本高清撒尿pissing| 无忧传媒在线观看视频| 亚洲综合另类精品小说| 人妻激情图片视频小说| 国产中文字幕四区在线观看| 国产精品视频男人的天堂| 一区二区三区四区视频| 日日夜夜大香蕉伊人| jiujiure精品视频在线| 久久久久五月天丁香社区| 久久尻中国美女视频| 97人人模人人爽人人喊| 一区二区三区美女毛片| 日本丰满熟妇BBXBBXHD| 中文乱理伦片在线观看| 天天日天天干天天舔天天射| 成熟熟女国产精品一区| 最近的中文字幕在线mv视频| 亚洲一级特黄特黄黄色录像片| 成年午夜影片国产片| 成人国产激情自拍三区| 色在线观看视频免费的| 国产夫妻视频在线观看免费| 国产麻豆精品人妻av| 91精品国产91青青碰| 久久久极品久久蜜桃| 大胆亚洲av日韩av| 国产成人综合一区2区| 中国产一级黄片免费视频播放| 欧美在线偷拍视频免费看| 色花堂在线av中文字幕九九| 日本熟女50视频免费| 美女福利视频网址导航| 人人爽亚洲av人人爽av| 国产aⅴ一线在线观看| 93精品视频在线观看| 亚洲一区自拍高清免费视频| av日韩在线免费播放| 啊啊好慢点插舔我逼啊啊啊视频| 午夜精品亚洲精品五月色| 亚洲精品av在线观看| 午夜精品久久久久久99热| 亚洲国产欧美一区二区三区…| 熟女人妻在线中出观看完整版| 福利一二三在线视频观看| 伊拉克及约旦宣布关闭领空| 国产白嫩美女一区二区| 久久久久久9999久久久久| 国产欧美精品不卡在线| 美洲精品一二三产区区别| 中国黄色av一级片| 精品suv一区二区69| 亚洲综合一区二区精品久久| 日本最新一二三区不卡在线 | 色呦呦视频在线观看视频| 最新国产亚洲精品中文在线| 欧美熟妇一区二区三区仙踪林| 在线可以看的视频你懂的| 天天日天天敢天天干| 日韩av大胆在线观看| 好吊操视频这里只有精品| 在线观看911精品国产| 婷婷激情四射在线观看视频| 久久久久只精品国产三级| 亚洲精品午夜久久久久| 黑人巨大精品欧美视频| 久草视频在线免播放| 日比视频老公慢点好舒服啊| 亚洲免费国产在线日韩| 国产精品久久9999| 初美沙希中文字幕在线| 国产午夜亚洲精品麻豆| 2020国产在线不卡视频 | 成人色综合中文字幕| 午夜在线观看一区视频| 久久久麻豆精亚洲av麻花| 美女福利写真在线观看视频| 日本免费午夜视频网站| av欧美网站在线观看| 18禁污污污app下载| 国产中文精品在线观看| 亚洲va欧美va人人爽3p| 午夜激情高清在线观看| 亚洲1069综合男同| 免费69视频在线看| 亚洲一区二区三区精品视频在线| 日本最新一二三区不卡在线 | 日韩成人综艺在线播放| 国产麻豆国语对白露脸剧情| 99热久久这里只有精品8| 色哟哟在线网站入口| av网址在线播放大全| 国产高清精品极品美女| 国产va精品免费观看| 91高清成人在线视频| 狠狠的往里顶撞h百合| 国产精品久久久久久久精品视频| 国产janese在线播放| 天天干夜夜操天天舔| 1000部国产精品成人观看视频| 日本女大学生的黄色小视频| 57pao国产一区二区| 日韩欧美制服诱惑一区在线| 日韩熟女系列一区二区三区| 中文字幕一区二区三区人妻大片 | 成年人该看的视频黄免费| 视频久久久久久久人妻| 视频一区二区三区高清在线| 五十路老熟女码av| 午夜dv内射一区区| 岛国av高清在线成人在线| 午夜影院在线观看视频羞羞羞| 亚洲午夜在线视频福利| 3337p日本欧洲大胆色噜噜| 夫妻在线观看视频91| 亚洲欧美激情国产综合久久久| 日韩二区视频一线天婷婷五| 97资源人妻免费在线视频| 欧美日韩人妻久久精品高清国产| 国产日本精品久久久久久久| 精品亚洲在线免费观看| 青青青青青青草国产| 国产免费高清视频视频| 77久久久久国产精产品| av在线免费观看亚洲天堂| 97精品视频在线观看| 日本女人一级免费片| 人妻av无码专区久久绿巨人| 老司机午夜精品视频资源 | 91 亚洲视频在线观看| 国产亚洲视频在线观看| 中出中文字幕在线观看| 色综合色综合色综合色| 中文字幕在线一区精品| 在线视频自拍第三页| 色97视频在线播放| 日韩视频一区二区免费观看| 午夜婷婷在线观看视频| 超碰在线中文字幕一区二区| 91在线免费观看成人| 中字幕人妻熟女人妻a62v网| 亚洲精品乱码久久久本| 亚洲天堂第一页中文字幕| 538精品在线观看视频| 天天干夜夜操啊啊啊| 国内精品在线播放第一页| 免费在线观看视频啪啪| 国产麻豆剧传媒精品国产av蜜桃 | 日本少妇人妻xxxxxhd| 影音先锋女人av噜噜色| 色哟哟国产精品入口| 日本少妇的秘密免费视频| 欧美精品久久久久久影院| 88成人免费av网站| 黄色视频成年人免费观看| 老司机免费福利视频网| 亚洲成人熟妇一区二区三区 | 婷婷综合亚洲爱久久| 蜜桃色婷婷久久久福利在线| 中文亚洲欧美日韩无线码| 91福利在线视频免费观看| 国产精品免费不卡av| 青青青青爽手机在线| 国产日韩av一区二区在线| 班长撕开乳罩揉我胸好爽| 亚洲在线观看中文字幕av| 成人福利视频免费在线| 五十路熟女人妻一区二区9933| 色花堂在线av中文字幕九九| 极品性荡少妇一区二区色欲| 欧美黑人与人妻精品| 日韩三级黄色片网站| 91she九色精品国产| 97人妻人人澡爽人人精品| 最新91精品视频在线| 国产丰满熟女成人视频| 巨乳人妻日下部加奈被邻居中出| 高潮喷水在线视频观看| 国产精品成人xxxx| 国产91嫩草久久成人在线视频| 天天日天天日天天射天天干 | 欧美男人大鸡吧插女人视频| 午夜蜜桃一区二区三区| 18禁美女羞羞免费网站| 一区二区在线观看少妇| 成人性爱在线看四区| 国产精品日韩欧美一区二区| 大香蕉玖玖一区2区| 天天摸天天干天天操科普| 超黄超污网站在线观看| 天天操天天干天天艹| 亚欧在线视频你懂的| 九色porny九色9l自拍视频| 操人妻嗷嗷叫视频一区二区| 国产欧美日韩在线观看不卡| 蜜臀成人av在线播放| 99热这里只有国产精品6| 亚洲最大黄 嗯色 操 啊| 日本xx片在线观看| 国产精彩福利精品视频| 亚洲嫩模一区二区三区| 最近中文字幕国产在线| 国产日韩一区二区在线看| 中文字幕国产专区欧美激情| 国产精品探花熟女在线观看| 成人24小时免费视频| 曰本无码人妻丰满熟妇啪啪| 欧洲精品第一页欧洲精品亚洲| 国产片免费观看在线观看| 全国亚洲男人的天堂| 国产欧美精品不卡在线| 伊拉克及约旦宣布关闭领空| 19一区二区三区在线播放| 91片黄在线观看喷潮| 日本一区精品视频在线观看| 经典亚洲伊人第一页| 亚洲最大黄 嗯色 操 啊| 黄片三级三级三级在线观看| 亚洲一区二区三区五区| 精品视频中文字幕在线播放| 欧美3p在线观看一区二区三区| 精品一区二区三区在线观看| 亚洲中文字字幕乱码| 老有所依在线观看完整版 | 亚洲人成精品久久久久久久| av日韩在线免费播放| 美女福利写真在线观看视频| 亚洲视频在线观看高清| 97年大学生大白天操逼| 偷拍自拍 中文字幕| 欧美老妇精品另类不卡片| 中文字幕在线一区精品| 99精品国产aⅴ在线观看| 91精品免费久久久久久| 欧美精品中文字幕久久二区| 午夜dv内射一区区| 亚洲综合在线视频可播放| 天天干夜夜操啊啊啊| 中文字幕1卡1区2区3区| 91国产在线免费播放| av乱码一区二区三区| 特大黑人巨大xxxx| 色偷偷伊人大杳蕉综合网 | 插逼视频双插洞国产操逼插洞| 岛国免费大片在线观看| 欧美在线精品一区二区三区视频| 亚洲熟色妇av日韩熟色妇在线| 欧美日韩不卡一区不区二区| 丰满少妇人妻xxxxx| 午夜在线精品偷拍一区二| 91欧美在线免费观看| 日韩精品电影亚洲一区| 色综合天天综合网国产成人| www,久久久,com| 97国产在线观看高清| 农村胖女人操逼视频| 日韩美在线观看视频黄| 午夜福利人人妻人人澡人人爽| 成人色综合中文字幕| 91 亚洲视频在线观看| 激情啪啪啪啪一区二区三区| 黄工厂精品视频在线观看| 国产之丝袜脚在线一区二区三区| 欧美精品久久久久久影院| 精品久久久久久久久久中文蒉| 亚洲欧美另类手机在线| 国产女人露脸高潮对白视频| 风流唐伯虎电视剧在线观看 | 久久久久久97三级| 欧美爆乳肉感大码在线观看| 91试看福利一分钟| 成人sm视频在线观看| 91老师蜜桃臀大屁股| 含骚鸡巴玩逼逼视频| 亚洲天堂av最新网址| 天堂中文字幕翔田av| 日日日日日日日日夜夜夜夜夜夜| 动漫黑丝美女的鸡巴| 在线免费91激情四射| 国产综合视频在线看片| 精品一区二区三区在线观看| 直接观看免费黄网站| 亚洲国产香蕉视频在线播放| 国产丰满熟女成人视频| 人妻爱爱 中文字幕| 蜜桃久久久久久久人妻| 2019av在线视频| 中文字幕第三十八页久久| 中文字幕综合一区二区| 人人妻人人爽人人澡人人精品| 都市激情校园春色狠狠| 日本韩国亚洲综合日韩欧美国产| 91天堂精品一区二区| 亚洲 自拍 色综合图| 欧美香蕉人妻精品一区二区| 国产又粗又黄又硬又爽| 亚洲 清纯 国产com| 亚洲图片偷拍自拍区| 亚洲精品在线资源站| 黄色的网站在线免费看| 亚洲欧美综合在线探花| 亚洲一区二区三区uij| 欧美一区二区中文字幕电影| 国产亚洲精品欧洲在线观看| 伊人开心婷婷国产av| 日本一区精品视频在线观看| 91天堂天天日天天操| 天天干天天日天天谢综合156| 亚洲午夜电影在线观看| 性感美女高潮视频久久久| 欧美成一区二区三区四区| 99久久中文字幕一本人| 亚洲精品国产在线电影| 亚洲国产精品免费在线观看| 亚洲偷自拍高清视频| 丰满的继坶3中文在线观看| 天干天天天色天天日天天射| 亚洲精品无码久久久久不卡| 999热精品视频在线| 蜜桃视频在线欧美一区| 黄色成年网站午夜在线观看| av中文字幕在线观看第三页| 天天干天天操天天摸天天射| 日韩二区视频一线天婷婷五| 国产成人一区二区三区电影网站| 中文字幕av男人天堂| 人妻爱爱 中文字幕| 99的爱精品免费视频| 66久久久久久久久久久| 亚洲国产免费av一区二区三区| 日韩熟女av天堂系列| 国产一区二区欧美三区| 色伦色伦777国产精品| 国内精品在线播放第一页| 国产亚洲成人免费在线观看| 国际av大片在线免费观看| av高潮迭起在线观看| 一级黄片大鸡巴插入美女| 在线观看欧美黄片一区二区三区| 欧美日韩激情啪啪啪| 中文字幕免费福利视频6| 老司机福利精品视频在线| 大陆胖女人与丈夫操b国语高清| 免费成人av中文字幕| 性色蜜臀av一区二区三区| 色花堂在线av中文字幕九九 | 91精品国产观看免费| 国产精品成人xxxx| 国产黄色高清资源在线免费观看| 中文字幕一区二区三区蜜月| 国产精品黄大片在线播放| 国产精品伦理片一区二区| 天天干天天爱天天色| 青青擦在线视频国产在线| 在线视频免费观看网| 亚洲中文字幕乱码区| 国产精品国产三级麻豆| 2019av在线视频| 亚洲视频乱码在线观看| 欧美日韩人妻久久精品高清国产| 欧洲欧美日韩国产在线| 亚洲精品中文字幕下载| 中文字幕av男人天堂| 老司机深夜免费福利视频在线观看| 国产刺激激情美女网站| 2019av在线视频| 99久久久无码国产精品性出奶水| 欧美日韩不卡一区不区二区| 亚洲 中文 自拍 无码| 亚洲专区激情在线观看视频| 人妻激情图片视频小说| 青草亚洲视频在线观看| 操人妻嗷嗷叫视频一区二区| 超级碰碰在线视频免费观看| 1区2区3区4区视频在线观看| 性感美女诱惑福利视频| 亚洲国产精品免费在线观看| 鸡巴操逼一级黄色气| 在线播放一区二区三区Av无码| 日本免费一级黄色录像| 成熟熟女国产精品一区| 国产精品人久久久久久| yellow在线播放av啊啊啊| 久久久制服丝袜中文字幕| 亚洲码av无色中文| h国产小视频福利在线观看| 久久精品国产23696| 75国产综合在线视频| 日韩欧美一级aa大片| 青娱乐蜜桃臀av色| 粉嫩av蜜乳av蜜臀| 又色又爽又黄的美女裸体| 亚洲成高清a人片在线观看| 精品人妻伦一二三区久| 抽查舔水白紧大视频| 久久热久久视频在线观看| 在线免费观看视频一二区| 日日夜夜精品一二三| 婷婷久久久久深爱网| 特级无码毛片免费视频播放| 超碰在线观看免费在线观看| 中文字幕人妻熟女在线电影| 欧美偷拍自拍色图片| 精品美女久久久久久| 日日爽天天干夜夜操| 午夜国产免费福利av| 天天射夜夜操综合网| 精品91自产拍在线观看一区| 欧美黑人与人妻精品| 91成人在线观看免费视频| 日韩av有码一区二区三区4| 成人区人妻精品一区二视频| 天天色天天操天天透| 性色蜜臀av一区二区三区| 午夜av一区二区三区 | 中文字幕日本人妻中出| 色婷婷六月亚洲综合香蕉| 超碰97人人澡人人| 亚洲一区二区三区五区| 天天干天天操天天爽天天摸| 国产成人精品久久二区91| 成人区人妻精品一区二视频| 亚洲激情唯美亚洲激情图片| 国产视频在线视频播放| 欧美va亚洲va天堂va| 中文亚洲欧美日韩无线码| 免费观看理论片完整版| 在线观看国产免费麻豆| 美女视频福利免费看| 国产亚州色婷婷久久99精品| 精品一区二区三区三区88 | 日本免费视频午夜福利视频| 男女第一次视频在线观看| 国产性生活中老年人视频网站| 黄色三级网站免费下载| 日韩二区视频一线天婷婷五| 久久这里只有精品热视频| 欧美黄色录像免费看的| 欧美精品 日韩国产| 中文字幕之无码色多多| 亚洲 清纯 国产com| 97超碰最新免费在线观看| 99精品视频之69精品视频| 熟女人妻在线观看视频| 中国老熟女偷拍第一页| 玖玖一区二区在线观看| 青青青激情在线观看视频| 日韩a级黄色小视频| 在线观看av2025| 中文字幕高清免费在线人妻| 亚洲欧美清纯唯美另类| 国产成人精品av网站| 亚洲1区2区3区精华液| 亚洲一区av中文字幕在线观看| 国产麻豆91在线视频| 成人亚洲国产综合精品| 久久午夜夜伦痒痒想咳嗽P| 2021国产一区二区| 国产女人露脸高潮对白视频| 欧美老鸡巴日小嫩逼| 中文字幕一区二区三区蜜月| 日本黄色特一级视频| 日本男女操逼视频免费看 | 日本脱亚入欧是指什么| 久久久久久97三级| 国产精彩福利精品视频| 欧美性受xx黑人性猛交| 人妻丝袜av在线播放网址| 18禁网站一区二区三区四区| 欧亚乱色一区二区三区| 亚洲av男人的天堂你懂的| 亚洲伊人久久精品影院一美女洗澡 | 日韩熟女系列一区二区三区| 无码中文字幕波多野不卡| 同居了嫂子在线播高清中文| 国产白袜脚足J棉袜在线观看| 中国把吊插入阴蒂的视频| 人妻另类专区欧美制服| 最新日韩av传媒在线| 国产精品人妻66p| 自拍偷拍一区二区三区图片 | 黑人大几巴狂插日本少妇| 男人的天堂av日韩亚洲| 大黑人性xxxxbbbb| 天天日天天摸天天爱| 国产亚洲精品视频合集| 国产精品自拍在线视频| 久久精品亚洲国产av香蕉| 超级碰碰在线视频免费观看| 姐姐的朋友2在线观看中文字幕| 亚洲欧美自拍另类图片| 亚洲一区二区三区精品乱码| 噜噜色噜噜噜久色超碰| 欧美色婷婷综合在线| 亚洲成人情色电影在线观看| 一区二区三区美女毛片| 同居了嫂子在线播高清中文| 99视频精品全部15| 黄色在线观看免费观看在线| 亚洲一区二区三区精品乱码| 美女张开腿让男生操在线看| 天天干天天爱天天色| 狠狠的往里顶撞h百合| 亚洲一区制服丝袜美腿 | 婷婷综合蜜桃av在线| 亚洲av日韩av第一区二区三区| 少妇被强干到高潮视频在线观看| 熟女少妇激情五十路| 二区中出在线观看老师| 亚洲一级美女啪啪啪| 日韩av有码一区二区三区4| 五十路av熟女松本翔子| 久久久久久久久久久免费女人| 成年人啪啪视频在线观看| 香港一级特黄大片在线播放| 国产综合视频在线看片| 国产精品污污污久久| 亚洲日产av一区二区在线| 国产精品视频男人的天堂| 欧美日本在线视频一区| 日本人妻精品久久久久久| 可以免费看的www视频你懂的| 亚洲卡1卡2卡三卡四老狼| 亚洲国产成人最新资源| 999久久久久999| 成人性黑人一级av| 亚洲一区二区三区五区| 性感美女高潮视频久久久| 521精品视频在线观看| 日本啪啪啪啪啪啪啪| 98视频精品在线观看| 日本高清成人一区二区三区| 亚洲欧美成人综合在线观看| 中文字日产幕乱六区蜜桃| 男人靠女人的逼视频| 亚洲av可乐操首页| 亚洲黄色av网站免费播放| 欧美激情电影免费在线| 一区二区在线观看少妇| 人妻激情图片视频小说| 青青青爽视频在线播放| 成年人免费看在线视频| 偷拍美女一区二区三区| 国产老熟女伦老熟妇ⅹ| 久久久久久99国产精品| av在线观看网址av| 欧美精品中文字幕久久二区| 中文字幕无码日韩专区免费| 国产精品女邻居小骚货| 非洲黑人一级特黄片| 亚洲欧美成人综合在线观看| 98视频精品在线观看| 成人av电影免费版| 高清成人av一区三区| 亚洲伊人久久精品影院一美女洗澡| 91成人精品亚洲国产| 丝袜国产专区在线观看| 亚洲在线观看中文字幕av| 国产精品探花熟女在线观看| okirakuhuhu在线观看| 天天干天天爱天天色| 中英文字幕av一区| 在线观看成人国产电影| 天天草天天色天天干| 香蕉91一区二区三区| 大香蕉伊人国产在线| 日本免费一级黄色录像| 男人的网址你懂的亚洲欧洲av| 久久精品美女免费视频| 国产日韩av一区二区在线| 91精品国产高清自在线看香蕉网 | 好了av中文字幕在线| 午夜毛片不卡免费观看视频| 粗大的内捧猛烈进出爽大牛汉子| 久久久精品国产亚洲AV一| 国产使劲操在线播放| 免费在线观看视频啪啪| 久精品人妻一区二区三区| 又色又爽又黄的美女裸体| 大香蕉大香蕉大香蕉大香蕉大香蕉| 男人在床上插女人视频| rct470中文字幕在线| 三上悠亚和黑人665番号| 亚洲偷自拍高清视频| 日比视频老公慢点好舒服啊| 绯色av蜜臀vs少妇| 一区二区免费高清黄色视频| 91色老99久久九九爱精品| 亚洲精品久久视频婷婷| 香港三日本三韩国三欧美三级| 五十路熟女人妻一区二区9933| 3344免费偷拍视频| 一色桃子人妻一区二区三区| 欧美一区二区三区乱码在线播放| 中文字幕亚洲久久久| 18禁精品网站久久| 2025年人妻中文字幕乱码在线| 二区中出在线观看老师| 欧美第一页在线免费观看视频| 午夜在线观看一区视频| 精品首页在线观看视频| 成人精品视频99第一页| 日本熟妇一区二区x x| 老熟妇凹凸淫老妇女av在线观看| 天天干狠狠干天天操| 青青草国内在线视频精选| 国产综合视频在线看片| 又色又爽又黄的美女裸体| 青娱乐极品视频青青草| wwwxxx一级黄色片| 色伦色伦777国产精品| 水蜜桃一区二区三区在线观看视频| 新97超碰在线观看| 国产伊人免费在线播放| 国产在线拍揄自揄视频网站| 国产av福利网址大全| 国产精品视频男人的天堂| 美女 午夜 在线视频| 黄色片黄色片wyaa| 国产精品探花熟女在线观看| 亚洲精品精品国产综合| 国产精品国产三级国产精东| 久久久噜噜噜久久熟女av| 日本又色又爽又黄又粗| 人妻无码色噜噜狠狠狠狠色| 午夜大尺度无码福利视频| 久草免费人妻视频在线| 桃色视频在线观看一区二区| 国产久久久精品毛片| 亚洲老熟妇日本老妇| 国产极品精品免费视频| 大陆胖女人与丈夫操b国语高清 | 风流唐伯虎电视剧在线观看| 亚洲av日韩精品久久久| 日本免费午夜视频网站| 亚洲免费va在线播放| 都市激情校园春色狠狠| 少妇被强干到高潮视频在线观看| 欧美在线偷拍视频免费看| 国产亚洲国产av网站在线| 日本高清撒尿pissing| 日本性感美女写真视频| 国产伦精品一区二区三区竹菊| 亚洲精品乱码久久久久久密桃明| 99精品国产aⅴ在线观看| 91色秘乱一区二区三区| 国产日韩精品电影7777| 直接能看的国产av| 亚洲精品欧美日韩在线播放 | 白嫩白嫩美女极品国产在线观看| 亚洲成人免费看电影| 爱爱免费在线观看视频| 天天干天天操天天摸天天射| 天天日夜夜操天天摸| 久青青草视频手机在线免费观看| 在线免费观看国产精品黄色| av中文字幕电影在线看| 天天爽夜夜爽人人爽QC| av手机在线观播放网站| 成人午夜电影在线观看 久久| 国产精品日韩欧美一区二区| 久久永久免费精品人妻专区 | 日韩欧美中文国产在线| 国产精品久久久久久美女校花| 91破解版永久免费| 亚洲一级 片内射视正片| 亚洲va国产va欧美va在线| 91免费福利网91麻豆国产精品| 亚洲天天干 夜夜操| 亚洲一区二区三区偷拍女厕91 | 日韩欧美国产一区不卡| 大鸡巴操b视频在线| 亚洲在线免费h观看网站| 青青青青青操视频在线观看| 国产精品免费不卡av| 真实国模和老外性视频| 日韩美女福利视频网| www天堂在线久久| 国产在线免费观看成人| 亚洲福利天堂久久久久久 | 亚洲粉嫩av一区二区三区| 中英文字幕av一区| 久久免看30视频口爆视频| 黄色三级网站免费下载| 99国内小视频在现欢看| 日本一区精品视频在线观看| 喷水视频在线观看这里只有精品| 韩国一级特黄大片做受| 天码人妻一区二区三区在线看| av日韩在线观看大全| 人妻久久久精品69系列| 在线播放 日韩 av| 91自产国产精品视频| 任我爽精品视频在线播放| 精品国产午夜视频一区二区| 久久丁香婷婷六月天| 国产精品久久综合久久| 精品人人人妻人人玩日产欧| 国产极品美女久久久久久| 欧美偷拍亚洲一区二区| 国产麻豆剧传媒精品国产av蜜桃 | 粉嫩av蜜乳av蜜臀 | 日韩精品二区一区久久| 国产chinesehd精品麻豆| 国产av欧美精品高潮网站| 日韩伦理短片在线观看| 欧美亚洲少妇福利视频| 97人人模人人爽人人喊 | 韩国三级aaaaa高清视频 | 中国黄色av一级片| 国产精品久久久久网| 欧美精品亚洲精品日韩在线| aiss午夜免费视频| 超pen在线观看视频公开97| 久久尻中国美女视频| 国产日韩欧美视频在线导航| 久久久久久9999久久久久| 天美传媒mv视频在线观看| 国产麻豆乱子伦午夜视频观看| 青青草亚洲国产精品视频| 亚洲国产在线精品国偷产拍| 午夜精品一区二区三区4| 精品区一区二区三区四区人妻| 黑人进入丰满少妇视频| 蜜桃精品久久久一区二区| 成年人午夜黄片视频资源| 色婷婷精品大在线观看| av在线shipin| 免费成人va在线观看| 国产在线自在拍91国语自产精品| 亚洲推理片免费看网站| 国产日韩精品一二三区久久久| 9l人妻人人爽人人爽| 99精品视频之69精品视频| 啪啪啪18禁一区二区三区| 天堂av在线官网中文| 又粗又硬又猛又黄免费30| 中文字幕av一区在线观看| 91九色国产porny蝌蚪| 伊人综合免费在线视频| 粗大的内捧猛烈进出爽大牛汉子| 熟女妇女老妇一二三区| 国产乱子伦一二三区| 99精品免费久久久久久久久a| 超鹏97历史在线观看| av中文字幕电影在线看| 欧美精品黑人性xxxx| 福利视频一区二区三区筱慧| 欧美另类z0z变态| 一区二区三区av高清免费| 国产又粗又黄又硬又爽| 亚洲色偷偷综合亚洲AV伊人| 午夜dv内射一区区| 在线网站你懂得老司机| 麻豆性色视频在线观看| 精品人妻每日一部精品| 91极品大一女神正在播放| 一区二区视频在线观看视频在线| 黑人巨大精品欧美视频| 天天日天天干天天舔天天射| 国产免费av一区二区凹凸四季| 日本成人一区二区不卡免费在线| 成年人黄色片免费网站| 中文字幕中文字幕人妻| 亚洲精品在线资源站| 激情国产小视频在线| 青青青视频自偷自拍38碰| 最新97国产在线视频| 日韩美女精品视频在线观看网站 | 人人妻人人澡欧美91精品 | 视频在线亚洲一区二区| 国产高清精品一区二区三区| 女同久久精品秋霞网| 亚洲 人妻 激情 中文| jiujiure精品视频在线| 十八禁在线观看地址免费| 日韩近亲视频在线观看| 狠狠的往里顶撞h百合| 亚洲美女高潮喷浆视频| 黄色视频在线观看高清无码| 午夜精品久久久久久99热| 无码日韩人妻精品久久| 婷婷综合蜜桃av在线| 亚洲成高清a人片在线观看| 亚洲av香蕉一区区二区三区犇| 一区二区三区av高清免费| 美女小视频网站在线| caoporn蜜桃视频| 亚洲欧美福利在线观看| 九九热99视频在线观看97| 天天操天天干天天日狠狠插| 青青青青视频在线播放| 中文字幕av男人天堂| 亚洲中文字幕人妻一区| 国产精品国产三级国产午| 精品久久婷婷免费视频| 国产品国产三级国产普通话三级| 最后99天全集在线观看| 欧洲日韩亚洲一区二区三区| 日本一二三区不卡无| 欧美亚洲国产成人免费在线| 中文字幕在线欧美精品| 中文字幕免费在线免费| 天天日天天干天天干天天日| 超级av免费观看一区二区三区| 日韩精品啪啪视频一道免费| 国产午夜亚洲精品不卡在线观看| 久久精品久久精品亚洲人| 韩国亚洲欧美超一级在线播放视频| 婷婷久久一区二区字幕网址你懂得 | 国产女人被做到高潮免费视频 | 亚洲精品成人网久久久久久小说| 国产精品三级三级三级| 欧美精品资源在线观看| 国产精品入口麻豆啊啊啊| 91chinese在线视频| 日韩av免费观看一区| 无码日韩人妻精品久久| 国产精品一二三不卡带免费视频| 激情综合治理六月婷婷| 在线成人日韩av电影| 午夜精品一区二区三区更新| 瑟瑟视频在线观看免费视频| 青娱乐最新视频在线| 青青在线视频性感少妇和隔壁黑丝| 99精品免费久久久久久久久a| 综合精品久久久久97| 午夜极品美女福利视频| 中文字幕人妻被公上司喝醉在线| 国产高潮无码喷水AV片在线观看| 日本一区美女福利视频| 亚洲成人免费看电影| 日韩av熟妇在线观看| 在线观看视频污一区| 100%美女蜜桃视频| 亚洲 清纯 国产com| 日本乱人一区二区三区| 国产真实灌醉下药美女av福利| 91p0rny九色露脸熟女| 国产亚洲精品视频合集| av破解版在线观看| 在线观看国产网站资源| 老师让我插进去69AV| 亚洲1区2区3区精华液| 国语对白xxxx乱大交| av天堂中文字幕最新| 91超碰青青中文字幕| 中文人妻AV久久人妻水| 日本免费午夜视频网站| 精品亚洲中文字幕av| 中文字幕在线观看极品视频| 亚洲自拍偷拍精品网| 一区二区三区四区五区性感视频| 18禁美女黄网站色大片下载| 日本韩国在线观看一区二区| 国产乱子伦一二三区| 欧洲精品第一页欧洲精品亚洲| 91色网站免费在线观看| 一区二区三区精品日本| 自拍偷拍亚洲欧美在线视频| 新97超碰在线观看| 亚洲图片偷拍自拍区| 亚洲人妻av毛片在线| 激情五月婷婷免费视频| 亚洲码av无色中文| 国产麻豆国语对白露脸剧情| 第一福利视频在线观看| 久久免看30视频口爆视频| 无码精品一区二区三区人| 亚洲激情偷拍一区二区| 鸡巴操逼一级黄色气| 91精品国产黑色丝袜| 日本少妇人妻xxxxx18| 最新国产亚洲精品中文在线| 国产成人精品av网站| 一本久久精品一区二区| 韩国三级aaaaa高清视频| 国产午夜福利av导航| 女生自摸在线观看一区二区三区 | 亚洲av香蕉一区区二区三区犇| 一级黄片大鸡巴插入美女| 2019av在线视频| 懂色av之国产精品| 涩涩的视频在线观看视频| 国产成人午夜精品福利| 一区二区免费高清黄色视频| 国产精品人妻66p| 久久免看30视频口爆视频| 黑人借宿ntr人妻的沦陷2| 亚洲综合另类精品小说| 爱有来生高清在线中文字幕| 韩国女主播精品视频网站| 97人妻总资源视频| 国产老熟女伦老熟妇ⅹ| 九九视频在线精品播放| 欧美乱妇无乱码一区二区| 天天操夜夜操天天操天天操| 天天做天天干天天操天天射| 亚洲中文精品人人免费| 阿v天堂2014 一区亚洲| av中文字幕在线观看第三页| av无限看熟女人妻另类av| 日本三极片视频网站观看| 亚洲欧美在线视频第一页| 欧美日本在线视频一区| 午夜久久香蕉电影网| 青青在线视频性感少妇和隔壁黑丝| 天天插天天色天天日| 91久久综合男人天堂| 骚货自慰被发现爆操| 综合激情网激情五月五月婷婷| 亚洲1069综合男同| 午夜精品福利一区二区三区p| 人妻久久无码中文成人| av破解版在线观看| 婷婷激情四射在线观看视频| 操的小逼流水的文章| 特级欧美插插插插插bbbbb| 大鸡吧插逼逼视频免费看| 国产精品黄色的av| 日韩精品二区一区久久| 日韩激情文学在线视频 | 天天操天天污天天射| 区一区二区三国产中文字幕| 欧美一级色视频美日韩| 亚洲精品乱码久久久本| 和邻居少妇愉情中文字幕| 熟女妇女老妇一二三区| 夜色撩人久久7777| 青青青视频手机在线观看| 亚洲福利精品福利精品福利| 亚洲欧美福利在线观看|