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

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

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

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

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

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

用戶文件格式:

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

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

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

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

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

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

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

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

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

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

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

4.FTP主類

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

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

總結:

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

相關文章

最新評論

青娱乐极品视频青青草| 成人网18免费视频版国产| 欧美韩国日本国产亚洲| 国产高清女主播在线| 最新国产亚洲精品中文在线| 亚洲女人的天堂av| 99精品久久久久久久91蜜桃| 色综合久久久久久久久中文| 成年人午夜黄片视频资源| 久久人人做人人妻人人玩精品vr| 888欧美视频在线| 国产男女视频在线播放| 日日夜夜狠狠干视频| 天堂av在线最新版在线| 99精品免费观看视频| 日日夜夜精品一二三| 中文字幕人妻av在线观看| 久久国产精品精品美女| 激情内射在线免费观看| 午夜av一区二区三区| 大黑人性xxxxbbbb| 日本又色又爽又黄又粗| 成人动漫大肉棒插进去视频| 综合激情网激情五月天| 99热久久极品热亚洲| 美女福利视频网址导航| 伊人精品福利综合导航| 涩爱综合久久五月蜜臀| 蜜桃视频入口久久久| 天天躁夜夜躁日日躁a麻豆| 青青青视频自偷自拍38碰| 91香蕉成人app下载| 最新国产精品网址在线观看| 国产福利小视频大全| 国产精品中文av在线播放| 91小伙伴中女熟女高潮| 国产成人精品av网站| 成人动漫大肉棒插进去视频| 男生舔女生逼逼视频| 精品久久久久久高潮| 欧美xxx成人在线| 欧美日韩人妻久久精品高清国产 | 鸡巴操逼一级黄色气| 欧美精品黑人性xxxx| 在线播放 日韩 av| 精品国产在线手机在线| 色伦色伦777国产精品| 久久久久久久精品成人热| 自拍偷拍日韩欧美亚洲| 2019av在线视频| 黄色男人的天堂视频| 亚洲欧美成人综合在线观看| 亚洲精品av在线观看| 边摸边做超爽毛片18禁色戒| 天天干夜夜操啊啊啊| 国产精品黄大片在线播放| 2021最新热播中文字幕| 亚洲一区二区三区久久午夜| 天天做天天爽夜夜做少妇| 日本三极片视频网站观看| 成年人啪啪视频在线观看| 成人午夜电影在线观看 久久| sspd152中文字幕在线| 国产高清97在线观看视频| 亚洲va国产va欧美va在线| 午夜精品久久久久久99热| 国产成人自拍视频播放| 色哟哟国产精品入口| 日本成人一区二区不卡免费在线| 天天躁日日躁狠狠躁躁欧美av| 中文字幕最新久久久| 国产三级片久久久久久久| 91精品高清一区二区三区| 直接观看免费黄网站| 不卡精品视频在线观看| 国产精品人久久久久久| 国产亚洲天堂天天一区| 中国黄色av一级片| 亚洲一级特黄特黄黄色录像片| 国产第一美女一区二区三区四区| 国产成人精品一区在线观看| 亚洲 自拍 色综合图| 丁香花免费在线观看中文字幕| 中英文字幕av一区| 视频在线亚洲一区二区| 人妻最新视频在线免费观看| 天堂av在线播放免费| aaa久久久久久久久| 亚洲欧美综合在线探花| 三级av中文字幕在线观看| 老有所依在线观看完整版| 嫩草aⅴ一区二区三区| 免费啪啪啪在线观看视频| 亚洲欧美激情人妻偷拍| 91精品综合久久久久3d动漫| 日本高清在线不卡一区二区| 粉嫩av蜜乳av蜜臀| 自拍偷拍亚洲欧美在线视频| 人人爽亚洲av人人爽av| 国产刺激激情美女网站| 91片黄在线观看喷潮| 亚洲欧美成人综合视频| 绯色av蜜臀vs少妇| 亚洲成人激情视频免费观看了| 欧美精品一二三视频| 亚洲午夜伦理视频在线| 久久久人妻一区二区| 三上悠亚和黑人665番号| 久久久久久97三级| 日韩影片一区二区三区不卡免费| 东京热男人的av天堂| 国产第一美女一区二区三区四区| 91天堂天天日天天操| 日韩欧美高清免费在线| 亚洲熟妇x久久av久久| 91亚洲手机在线视频播放| 亚洲欧美精品综合图片小说| 国产成人精品久久二区91| 久久精品国产23696| lutube在线成人免费看| 亚欧在线视频你懂的| 一区二区熟女人妻视频| 扒开腿挺进肉嫩小18禁视频| 99视频精品全部15| 999九九久久久精品| 婷婷综合蜜桃av在线| 亚洲一区二区三区久久午夜 | 粉嫩小穴流水视频在线观看| 夏目彩春在线中文字幕| 国产av欧美精品高潮网站| 国产乱子伦精品视频潮优女| 欧美特级特黄a大片免费| 午夜激情精品福利视频| 日韩少妇人妻精品无码专区| 91精品一区二区三区站长推荐| 亚洲精品午夜久久久久| 可以免费看的www视频你懂的| 日本韩国免费福利精品| 成人性爱在线看四区| 国产精品自偷自拍啪啪啪| 日韩一区二区三区三州| 99热99这里精品6国产| 蜜桃视频入口久久久| lutube在线成人免费看| 91国内视频在线观看| 欧美亚洲牲夜夜综合久久| 蜜桃视频17c在线一区二区| 国产中文精品在线观看| 亚洲欧美精品综合图片小说| 自拍偷拍亚洲欧美在线视频| 欧美va不卡视频在线观看| 久久午夜夜伦痒痒想咳嗽P| 天天操天天干天天日狠狠插| 中国视频一区二区三区| 久久久91蜜桃精品ad| 欧美黑人性暴力猛交喷水| 久久丁香花五月天色婷婷| 一区二区三区日本伦理| 欧洲日韩亚洲一区二区三区 | 男人的天堂在线黄色| mm131美女午夜爽爽爽| 任你操视频免费在线观看| 在线观看视频 你懂的| 久久久久久国产精品| 亚洲成人激情视频免费观看了| 亚洲国产免费av一区二区三区 | 91色九色porny| 欧美亚洲国产成人免费在线 | 欧美一区二区三区乱码在线播放| 日本阿v视频在线免费观看| 日韩中文字幕精品淫| 午夜频道成人在线91| 亚洲欧美成人综合在线观看| 在线免费观看黄页视频| 国产又大又黄免费观看| 天天艹天天干天天操| 91在线免费观看成人| 日视频免费在线观看| 精产国品久久一二三产区区别| 午夜精品福利一区二区三区p | 十八禁在线观看地址免费 | 91天堂精品一区二区| 天天操天天射天天操天天天| 青青尤物在线观看视频网站| 中文字幕综合一区二区| 春色激情网欧美成人| 天天躁日日躁狠狠躁av麻豆| 亚洲综合在线视频可播放| 成人区人妻精品一区二视频| 国产一区二区在线欧美| 中文字幕 码 在线视频| 亚洲一区二区三区精品乱码| 久久久久久久亚洲午夜综合福利| 内射久久久久综合网| 中英文字幕av一区| av在线观看网址av| 98精产国品一二三产区区别| 日韩中文字幕精品淫| 78色精品一区二区三区| 成人色综合中文字幕| 天天摸天天干天天操科普| 青春草视频在线免费播放| 精品国产亚洲av一淫| 亚洲精品 日韩电影| 亚洲国产成人在线一区| 中文字幕1卡1区2区3区| 亚洲国产在人线放午夜| 一个人免费在线观看ww视频| 五十路人妻熟女av一区二区| 国产剧情演绎系列丝袜高跟| 视频久久久久久久人妻| 午夜精品一区二区三区福利视频| 人妻少妇亚洲精品中文字幕| 淫秽激情视频免费观看| 免费在线观看污污视频网站| 极品粉嫩小泬白浆20p主播| 黄片三级三级三级在线观看| 亚洲欧美成人综合视频| 日韩精品中文字幕福利| 黄页网视频在线免费观看 | 亚洲熟妇久久无码精品| 伊人开心婷婷国产av| 久青青草视频手机在线免费观看| 午夜精品久久久久久99热| 中文字幕之无码色多多| 亚洲熟妇x久久av久久| 一色桃子久久精品亚洲| 久久精品国产999| 日韩欧美亚洲熟女人妻| 91久久人澡人人添人人爽乱| 亚洲第一伊人天堂网| 在线免费观看日本片| 性色蜜臀av一区二区三区| 欧美成人小视频在线免费看| 综合激情网激情五月天| 天天日天天干天天要| 55夜色66夜色国产精品站| 久久亚洲天堂中文对白| 搡老妇人老女人老熟女| 日韩美女搞黄视频免费| 中国黄色av一级片| 91色秘乱一区二区三区| 日本女人一级免费片| 3337p日本欧洲大胆色噜噜| 成人国产影院在线观看| 亚洲麻豆一区二区三区| 19一区二区三区在线播放| 天天干天天啪天天舔| 2021久久免费视频| 很黄很污很色的午夜网站在线观看| 国产亚洲欧美另类在线观看| 精品一区二区三区午夜| 黄网十四区丁香社区激情五月天| 狠狠躁狠狠爱网站视频| 日本一区二区三区免费小视频| 五色婷婷综合狠狠爱| 2019av在线视频| 亚洲女人的天堂av| 日韩特级黄片高清在线看| 人人在线视频一区二区| 亚洲最大黄了色网站| eeuss鲁片一区二区三区| 亚洲一区二区久久久人妻| 中文字幕 码 在线视频| 青青青视频手机在线观看| 伊人开心婷婷国产av| 91色秘乱一区二区三区| 男人天堂最新地址av| 成人sm视频在线观看| 男人和女人激情视频| 蜜桃久久久久久久人妻| 免费啪啪啪在线观看视频| 亚洲av无硬久久精品蜜桃| 中文字幕人妻熟女在线电影| 日本a级视频老女人| 男人的网址你懂的亚洲欧洲av| 骚逼被大屌狂草视频免费看| 蜜桃色婷婷久久久福利在线| 亚洲综合在线视频可播放| 欧美交性又色又爽又黄麻豆| 国产污污污污网站在线| 97国产福利小视频合集| av乱码一区二区三区| 国产白嫩美女一区二区| 91精品国产综合久久久蜜| av网站色偷偷婷婷网男人的天堂| 日本脱亚入欧是指什么| 任你操视频免费在线观看| 88成人免费av网站| 在线视频国产欧美日韩| 成年美女黄网站18禁久久| 成人在线欧美日韩国产| 久久这里有免费精品| 极品丝袜一区二区三区| 国产精品视频男人的天堂| 中文字幕日韩无敌亚洲精品| 少妇被强干到高潮视频在线观看| 欧美性感尤物人妻在线免费看| 夜夜嗨av蜜臀av| 天天日天天干天天爱| 国产+亚洲+欧美+另类| 久久www免费人成一看片| 欧美另类重口味极品在线观看| 久久麻豆亚洲精品av| 九九视频在线精品播放| 国产综合视频在线看片| 亚洲视频乱码在线观看| 亚洲一区二区三区在线高清| aⅴ精产国品一二三产品| 亚洲精品国产久久久久久| 亚洲欧美一区二区三区电影| 欧洲精品第一页欧洲精品亚洲| 在线国产日韩欧美视频| 欧美一区二区三区高清不卡tv | av中文字幕国产在线观看| 免费看国产av网站| 在线播放一区二区三区Av无码| 国产使劲操在线播放| 亚洲天堂第一页中文字幕| 中国视频一区二区三区| av线天堂在线观看| 国产中文精品在线观看| 亚洲熟妇久久无码精品| 午夜精品久久久久久99热| 老司机午夜精品视频资源| 女同性ⅹxx女同h偷拍| 91九色国产熟女一区二区| 亚洲综合乱码一区二区| 国产一级麻豆精品免费| 国产日本精品久久久久久久| 中文字幕高清在线免费播放| 亚洲av日韩精品久久久久久hd| 偷拍自拍亚洲美腿丝袜| 操的小逼流水的文章| 欧美色婷婷综合在线| 亚洲成人午夜电影在线观看| 欧美交性又色又爽又黄麻豆| 青青尤物在线观看视频网站| 美女av色播在线播放| a v欧美一区=区三区| 亚洲欧美清纯唯美另类| av手机在线观播放网站| 五十路av熟女松本翔子| 亚洲av色香蕉一区二区三区| 黄色片一级美女黄色片| 动漫美女的小穴视频| 久草免费人妻视频在线| av在线资源中文字幕| 国产麻豆乱子伦午夜视频观看| 国产女孩喷水在线观看| 欧美aa一级一区三区四区| 亚洲免费在线视频网站| 日本高清撒尿pissing| 骚货自慰被发现爆操| 亚洲人妻30pwc| 国产亚洲欧美另类在线观看| 动漫美女的小穴视频| 色秀欧美视频第一页| 欧美成人小视频在线免费看| 欧洲亚洲欧美日韩综合| 欧美另类重口味极品在线观看| 伊人网中文字幕在线视频| 久久一区二区三区人妻欧美| 精品老妇女久久9g国产| 88成人免费av网站| 51国产偷自视频在线播放| 蜜桃视频在线欧美一区| av乱码一区二区三区| 福利在线视频网址导航| 午夜在线观看一区视频| 人妻3p真实偷拍一二区| 嫩草aⅴ一区二区三区| 福利午夜视频在线合集| 亚洲蜜臀av一区二区三区九色| 亚洲av午夜免费观看| 激情人妻校园春色亚洲欧美 | 午夜的视频在线观看| 久草电影免费在线观看| 91在线视频在线精品3| 啊啊好大好爽啊啊操我啊啊视频| 天天日天天爽天天爽| 在线免费91激情四射| 亚洲欧美一区二区三区爱爱动图| 国产在线观看免费人成短视频| 青青青青青青青青青国产精品视频| 2020久久躁狠狠躁夜夜躁| 欧美精品一二三视频| 久久永久免费精品人妻专区 | 99久久99一区二区三区| 中文字幕一区二区三区蜜月| 大香蕉福利在线观看| 亚洲精品高清自拍av | 国产精品久久综合久久| 大学生A级毛片免费视频| 熟妇一区二区三区高清版| 性欧美日本大妈母与子| 亚洲精品三级av在线免费观看| 日韩av有码中文字幕| 亚洲国产美女一区二区三区软件 | 夜夜骑夜夜操夜夜奸| 国产又粗又猛又爽又黄的视频美国| 40道精品招牌菜特色| 超碰97人人做人人爱| 天天干夜夜操天天舔| 欲乱人妻少妇在线视频裸| 超黄超污网站在线观看| 亚洲变态另类色图天堂网| 热思思国产99re| 97国产在线av精品| 强行扒开双腿猛烈进入免费版| 91桃色成人网络在线观看| 日韩美女福利视频网| 国产内射中出在线观看| 欧美成人综合色在线噜噜| 亚洲中文字幕综合小综合| 国产中文精品在线观看| 开心 色 六月 婷婷| 五十路熟女人妻一区二区9933| 亚洲国产成人在线一区| 老司机福利精品免费视频一区二区 | 久久久久久久亚洲午夜综合福利| 一区国内二区日韩三区欧美| 久久久制服丝袜中文字幕| 久久国产精品精品美女| 国产91精品拍在线观看| 亚洲视频在线观看高清| 日本www中文字幕| 欧美第一页在线免费观看视频| 在线观看免费视频色97| 夜色撩人久久7777| 欧美黄片精彩在线免费观看 | 新97超碰在线观看| 馒头大胆亚洲一区二区| lutube在线成人免费看| 97年大学生大白天操逼| 中文字幕AV在线免费看 | 天天日天天日天天擦| 青娱乐在线免费视频盛宴| 日本精品美女在线观看| 最新激情中文字幕视频| 国产精品一区二区av国| 色婷婷久久久久swag精品| 国产精品视频资源在线播放| 国产亚洲精品品视频在线| 国产免费av一区二区凹凸四季| 精品成人午夜免费看| 啊用力插好舒服视频| 亚洲激情,偷拍视频| 免费成人av中文字幕| 五十路av熟女松本翔子| 亚洲精品国偷自产在线观看蜜桃| 阴茎插到阴道里面的视频| 啪啪啪啪啪啪啪免费视频| 日韩北条麻妃一区在线| 欧美久久一区二区伊人| 姐姐的朋友2在线观看中文字幕| 99热久久极品热亚洲| 亚洲午夜电影在线观看| 欧美亚洲一二三区蜜臀| 亚洲区欧美区另类最新章节| 国产日韩欧美美利坚蜜臀懂色| 久草极品美女视频在线观看| 国产午夜亚洲精品不卡在线观看| 青娱乐蜜桃臀av色| 午夜在线观看岛国av,com| 国产亚洲精品欧洲在线观看| 日日夜夜大香蕉伊人| 免费黄色成人午夜在线网站| 极品粉嫩小泬白浆20p主播| 中文字幕第一页国产在线| 日本高清撒尿pissing| 亚洲国产欧美一区二区三区久久 | 国产精品人妻熟女毛片av久| 亚洲嫩模一区二区三区| 激情综合治理六月婷婷| 亚洲国产美女一区二区三区软件| 精品乱子伦一区二区三区免费播 | 亚洲 清纯 国产com| 亚洲一级av大片免费观看| 中文字幕无码日韩专区免费| 777奇米久久精品一区| 爱有来生高清在线中文字幕| av线天堂在线观看| 91精品激情五月婷婷在线| 亚洲卡1卡2卡三卡四老狼| 在线视频自拍第三页| 亚洲第17页国产精品| 最新97国产在线视频| 18禁美女无遮挡免费| av新中文天堂在线网址| 国产精品精品精品999| av俺也去在线播放| 一区二区三区四区五区性感视频 | mm131美女午夜爽爽爽| 小穴多水久久精品免费看| 亚洲综合乱码一区二区| 亚洲欧美一区二区三区电影| 天天日天天干天天舔天天射| 日本少妇的秘密免费视频| 一区二区三区日本伦理| 天天通天天透天天插| 在线免费视频 自拍| 国产欧美精品免费观看视频| 中文字幕人妻三级在线观看| 搡老熟女一区二区在线观看| 国产免费高清视频视频| 亚洲特黄aaaa片| 欧美第一页在线免费观看视频| 国产在线91观看免费观看| 亚洲天堂精品福利成人av| 一区二区三区国产精选在线播放| 欧美精品久久久久久影院| 午夜福利资源综合激情午夜福利资| 懂色av之国产精品| 国产成人午夜精品福利| 国产日韩欧美美利坚蜜臀懂色| 天天干天天日天天干天天操| 色婷婷久久久久swag精品| 青青青青爽手机在线| 日韩精品激情在线观看| 国产精彩福利精品视频| 青青草成人福利电影| 亚洲精品午夜久久久久| 2025年人妻中文字幕乱码在线| 国产麻豆国语对白露脸剧情 | 晚上一个人看操B片| 五月天久久激情视频| 啊啊好大好爽啊啊操我啊啊视频| 乱亲女秽乱长久久久| 欧美在线一二三视频| 国产精品欧美日韩区二区| 把腿张开让我插进去视频| 男人插女人视频网站| 婷婷六月天中文字幕| 青草亚洲视频在线观看| 日韩午夜福利精品试看| 日韩欧美一级aa大片| 大尺度激情四射网站| 一区二区三区 自拍偷拍| 亚洲人一区二区中文字幕| 六月婷婷激情一区二区三区| 中文字幕人妻三级在线观看| 一区二区久久成人网| 欧美一级片免费在线成人观看| 亚洲伊人av天堂有码在线| 日韩美av高清在线| 日本韩国亚洲综合日韩欧美国产| 国产高清精品极品美女| 午夜福利资源综合激情午夜福利资| 亚洲综合在线视频可播放| 欧美xxx成人在线| 欧美成人小视频在线免费看| 亚洲图片偷拍自拍区| lutube在线成人免费看| 美洲精品一二三产区区别| 深田咏美亚洲一区二区| 午夜91一区二区三区| 色综合久久五月色婷婷综合| 午夜在线观看一区视频| 黄色av网站免费在线| av俺也去在线播放| 国产成人精品亚洲男人的天堂| 国产视频网站一区二区三区| 国产女人被做到高潮免费视频 | 中文字幕av熟女人妻| 欧美一区二区三区啪啪同性| 精品国产污污免费网站入口自 | 美洲精品一二三产区区别| aaa久久久久久久久| 亚洲一区二区久久久人妻| 最新国产精品拍在线观看| 欧美日本在线视频一区| 亚洲欧美久久久久久久久| 免费手机黄页网址大全| 日本少妇的秘密免费视频| 亚洲2021av天堂| 57pao国产一区二区| 免费岛国喷水视频在线观看| 香港一级特黄大片在线播放| 亚洲人人妻一区二区三区| 东京热男人的av天堂| 国产亚洲国产av网站在线| 18禁免费av网站| 亚洲国产第一页在线观看| 日韩人妻xxxxx| 男人在床上插女人视频| 日韩精品中文字幕福利| 日本熟妇喷水xxx| 日韩av有码中文字幕| 女生被男生插的视频网站| 99国内小视频在现欢看| 久久麻豆亚洲精品av| 青青青视频手机在线观看| 国产污污污污网站在线| 国产高清97在线观看视频| 老司机99精品视频在线观看| 国产麻豆剧传媒精品国产av蜜桃| 国产三级精品三级在线不卡| 久久h视频在线观看| 欧美一区二区三区高清不卡tv| 最新国产亚洲精品中文在线| 久久农村老妇乱69系列| 91亚洲精品干熟女蜜桃频道| 天美传媒mv视频在线观看| 欧美色婷婷综合在线| 欧美成人一二三在线网| 精品av久久久久久久| 天天日天天干天天插舔舔| 亚洲国产成人在线一区| 中文字幕日韩精品就在这里| 天天日天天天天天天天天天天| 黄色在线观看免费观看在线| 久久尻中国美女视频| 人妻久久无码中文成人| 成人综合亚洲欧美一区| 中文字幕一区二 区二三区四区| 国产九色91在线视频| 亚洲成人免费看电影| av在线资源中文字幕| 六月婷婷激情一区二区三区| 日韩av有码一区二区三区4| 亚洲成人情色电影在线观看 | av中文字幕在线观看第三页| 男女啪啪视频免费在线观看| 国产成人午夜精品福利| 精品高跟鞋丝袜一区二区| 夜夜嗨av蜜臀av| 国产福利小视频二区| 日本性感美女视频网站| 在线观看视频一区麻豆| 女蜜桃臀紧身瑜伽裤| 在线视频这里只有精品自拍| 在线观看免费视频网| 中文字幕AV在线免费看 | 99精品一区二区三区的区| 中文字幕成人日韩欧美| 成年人免费看在线视频| 中文字幕在线视频一区二区三区| 日韩a级精品一区二区| 色婷婷综合激情五月免费观看| 日比视频老公慢点好舒服啊| 精品亚洲在线免费观看| 亚洲免费视频欧洲免费视频| 亚洲精品在线资源站| 欧美精品资源在线观看| 大屁股肉感人妻中文字幕在线| 亚洲欧美一区二区三区爱爱动图| 中文字幕 亚洲av| 亚洲综合另类精品小说| 亚洲精品中文字幕下载| 色哟哟在线网站入口| 国产精品人妻熟女毛片av久| 东京热男人的av天堂| 亚洲的电影一区二区三区| 91精品国产91久久自产久强| 国产女人叫床高潮大片视频| 天天干天天操天天爽天天摸| 国产在线自在拍91国语自产精品| 97国产在线观看高清| 欧美日韩亚洲国产无线码| 天天射,天天操,天天说| 人妻少妇亚洲一区二区| 国产黄色大片在线免费播放| 一区二区三区的久久的蜜桃的视频 | 国产熟妇一区二区三区av| 中文字幕在线第一页成人| 在线视频免费观看网| 啪啪啪啪啪啪啪啪av| 9l人妻人人爽人人爽| 美女张开两腿让男人桶av| 日韩美女福利视频网| 成人影片高清在线观看| 欧美一区二区三区乱码在线播放| 我想看操逼黄色大片| 国产又色又刺激在线视频| 在线观看免费视频网| 91成人在线观看免费视频| 97资源人妻免费在线视频| 欧洲精品第一页欧洲精品亚洲| 国产夫妻视频在线观看免费| 男女啪啪视频免费在线观看| 在线不卡成人黄色精品| 社区自拍揄拍尻屁你懂的| 日本高清成人一区二区三区| 亚洲精品在线资源站| 超级av免费观看一区二区三区| 婷婷综合亚洲爱久久| 中文字幕高清在线免费播放| 狍和女人的王色毛片| 日本黄色三级高清视频| 丰满的子国产在线观看| 91‖亚洲‖国产熟女| 五十路人妻熟女av一区二区| 在线观看免费av网址大全| 男女啪啪视频免费在线观看| 国产精品成人xxxx| 中文字幕奴隷色的舞台50| 91极品新人『兔兔』精品新作| 国产精品亚洲在线观看| 青青社区2国产视频| 经典国语激情内射视频| 青青青视频手机在线观看| 日韩欧美亚洲熟女人妻| 福利在线视频网址导航| 91免费福利网91麻豆国产精品 | 成人av电影免费版| 早川濑里奈av黑人番号| 色综合久久久久久久久中文| 国产一区二区视频观看| 啊啊啊视频试看人妻| av在线播放国产不卡| 日韩亚国产欧美三级涩爱| 最新91精品视频在线| 亚洲成高清a人片在线观看| 日韩欧美高清免费在线| 美女福利视频导航网站| 日韩剧情片电影在线收看| 五十路老熟女码av| 5528327男人天堂| 亚洲一区二区三区偷拍女厕91| 天堂v男人视频在线观看| 天天操天天操天天碰| 亚洲最大黄了色网站| 国产精品伦理片一区二区| 在线免费观看国产精品黄色| 男人和女人激情视频| 欧美日韩高清午夜蜜桃大香蕉| 欲满人妻中文字幕在线| 顶级尤物粉嫩小尤物网站| 国产在线91观看免费观看| 国产V亚洲V天堂无码欠欠| 精品一区二区三区三区88| 亚洲视频乱码在线观看| 秋霞午夜av福利经典影视| 啊啊好大好爽啊啊操我啊啊视频 | 日本后入视频在线观看| 亚洲午夜伦理视频在线| 91大神福利视频网| 精品高跟鞋丝袜一区二区| 超级福利视频在线观看| 91麻豆精品传媒国产黄色片| 日本a级视频老女人| 一二三区在线观看视频| 免费在线福利小视频| 99久久成人日韩欧美精品| 2020中文字幕在线播放| av中文字幕网址在线| 精品高跟鞋丝袜一区二区| 久久久91蜜桃精品ad| 欧美视频不卡一区四区| 国产精品系列在线观看一区二区| 日韩av熟妇在线观看| 天天艹天天干天天操| 亚洲天堂精品久久久| 亚洲av天堂在线播放| 亚洲图片偷拍自拍区| 欧美怡红院视频在线观看| 777奇米久久精品一区| 黄色av网站免费在线| 乱亲女秽乱长久久久| 青青色国产视频在线| 国产黄色高清资源在线免费观看| 操人妻嗷嗷叫视频一区二区| 麻豆性色视频在线观看| 在线观看911精品国产| 2022国产综合在线干| 99精品视频在线观看婷婷| 韩国女主播精品视频网站| 精产国品久久一二三产区区别| 超黄超污网站在线观看| 九色视频在线观看免费| 久久久久久久久久一区二区三区| 亚洲成av人无码不卡影片一| 精品视频中文字幕在线播放| 国产在线免费观看成人| 欧美一区二区三区高清不卡tv| 超碰在线中文字幕一区二区| 伊人情人综合成人久久网小说| 中文字母永久播放1区2区3区| 青青热久免费精品视频在线观看| 91九色国产porny蝌蚪| 中文字幕乱码人妻电影| 精品日产卡一卡二卡国色天香| 人妻久久久精品69系列| 成人久久精品一区二区三区| 成人sm视频在线观看| 免费av岛国天堂网站| 日韩一区二区电国产精品| 亚洲综合自拍视频一区| 国产又粗又硬又大视频| 喷水视频在线观看这里只有精品| 日韩精品中文字幕播放| 在线免费观看视频一二区| 天天色天天爱天天爽| 欧洲黄页网免费观看| 欧美成人一二三在线网| 污污小视频91在线观看| 18禁美女黄网站色大片下载| 国产在线一区二区三区麻酥酥| 久久久久久久久久久免费女人| 一区二区视频在线观看免费观看 | 日韩近亲视频在线观看| 同居了嫂子在线播高清中文| 久久三久久三久久三久久| 一区二区三区另类在线 | 91久久国产成人免费网站| 538精品在线观看视频| 国产日韩欧美视频在线导航| 久久精品国产999| 亚洲区欧美区另类最新章节| 欧美黑人性猛交xxxxⅹooo| 熟女人妻在线观看视频| 午夜在线观看一区视频| 欧美精产国品一二三区| 中文字幕亚洲久久久| 特一级特级黄色网片| 天天射夜夜操狠狠干| 日本一区二区三区免费小视频| 国产伊人免费在线播放| aⅴ精产国品一二三产品| av在线观看网址av| 沙月文乃人妻侵犯中文字幕在线 | 国产精品欧美日韩区二区| yy96视频在线观看| 中英文字幕av一区| 免费在线观看污污视频网站| 一区二区三区四区五区性感视频| 一区二区三区在线视频福利| 色哟哟在线网站入口| 好吊视频—区二区三区| 国产精品自拍视频大全| 五十路熟女人妻一区二区9933| 一区二区视频在线观看免费观看| 亚洲男人在线天堂网| 最新日韩av传媒在线| 黑人解禁人妻叶爱071| 懂色av蜜桃a v| 亚洲va欧美va人人爽3p| 91社福利《在线观看| 亚洲专区激情在线观看视频| 天天日天天干天天爱| 国产精品视频资源在线播放| 51精品视频免费在线观看| 亚洲精品久久综合久| 红杏久久av人妻一区| 北条麻妃av在线免费观看| 亚洲av无码成人精品区辽| 国产真实灌醉下药美女av福利| 九九视频在线精品播放| 亚洲色偷偷综合亚洲AV伊人| 91天堂精品一区二区| 精品成人午夜免费看| 在线观看视频网站麻豆| 大鸡八强奸视频在线观看| 在线观看免费岛国av| 日韩伦理短片在线观看| 中文字幕一区二 区二三区四区| 唐人色亚洲av嫩草| 五月婷婷在线观看视频免费| 久久久久五月天丁香社区 | 一级a看免费观看网站| 91精品啪在线免费| 国产女人露脸高潮对白视频| 国产精品三级三级三级| 2018最新中文字幕在线观看| 视频一区二区综合精品| 熟女人妻在线观看视频| 天天做天天干天天舔| 亚洲丝袜老师诱惑在线观看| 国产女孩喷水在线观看| 精品亚洲中文字幕av| 免费大片在线观看视频网站| 日韩欧美国产一区不卡| 操操网操操伊剧情片中文字幕网| 在线新三级黄伊人网| 亚洲免费va在线播放| 懂色av之国产精品| 91色网站免费在线观看| 久久99久久99精品影院| 嫩草aⅴ一区二区三区| 最近中文2019年在线看| 性感美女诱惑福利视频| 国产亚洲精品品视频在线| 国产揄拍高清国内精品对白| 国产精品久久久久国产三级试频| 任你操视频免费在线观看| 日韩av有码一区二区三区4 | 91传媒一区二区三区| 国产精品视频欧美一区二区| 香蕉片在线观看av| 午夜在线一区二区免费| 成人蜜臀午夜久久一区| 韩国亚洲欧美超一级在线播放视频| 青青青青视频在线播放| 啪啪啪啪啪啪啪免费视频| 青青擦在线视频国产在线| 在线视频国产欧美日韩| 中文字幕高清在线免费播放| 91av中文视频在线| 青娱乐最新视频在线| 亚洲va国产va欧美va在线| 欧美国产亚洲中英文字幕| 久久久极品久久蜜桃| 久草极品美女视频在线观看| 中文字幕日韩人妻在线三区| 深田咏美亚洲一区二区| 91综合久久亚洲综合| 国产精品手机在线看片| 91在线免费观看成人| 偷拍自拍视频图片免费| 日本三极片视频网站观看| 免费观看理论片完整版| 精品高跟鞋丝袜一区二区| 天堂av中文在线最新版| 青青草原色片网站在线观看| 国产欧美精品一区二区高清| 热久久只有这里有精品| 久久久久五月天丁香社区| 精品成人午夜免费看| 日本成人一区二区不卡免费在线| 九色视频在线观看免费| 成人18禁网站在线播放| 成年人啪啪视频在线观看| 国产黄色a级三级三级三级| 中文字幕中文字幕人妻| 91she九色精品国产| 亚洲中文字幕乱码区| 边摸边做超爽毛片18禁色戒 | 日本乱人一区二区三区| 人人爽亚洲av人人爽av| 国产视频网站国产视频| 男人在床上插女人视频| ka0ri在线视频| 一区二区三区 自拍偷拍| 黄色视频成年人免费观看| 蜜桃视频入口久久久| 2020久久躁狠狠躁夜夜躁| 91欧美在线免费观看| 欧美日本在线视频一区| 国产一级麻豆精品免费| 2018最新中文字幕在线观看| 色哟哟国产精品入口| 2o22av在线视频| 视频在线亚洲一区二区| 中文字幕午夜免费福利视频| 99的爱精品免费视频| 888欧美视频在线| 大鸡吧插逼逼视频免费看 | 在线观看国产免费麻豆| 亚洲欧美综合在线探花| 成熟丰满熟妇高潮xx×xx| 一区二区免费高清黄色视频| 91精品综合久久久久3d动漫| 91国语爽死我了不卡| 在线播放一区二区三区Av无码| 亚洲高清国产自产av| 欧美一区二区中文字幕电影| 99精品国产aⅴ在线观看| 日韩三级黄色片网站| 亚洲特黄aaaa片| 国产视频在线视频播放| 国产一区二区三免费视频| 少妇高潮无套内谢麻豆| 国产91精品拍在线观看| 黄色的网站在线免费看| 日美女屁股黄邑视频| 中国视频一区二区三区| 在线可以看的视频你懂的| 欧美专区第八页一区在线播放| 亚洲老熟妇日本老妇| 中出中文字幕在线观看| 国产三级影院在线观看| 中文字幕,亚洲人妻| 蜜臀成人av在线播放| 人妻在线精品录音叫床| 午夜在线观看一区视频| 午夜精品一区二区三区福利视频| 激情伦理欧美日韩中文字幕| 5528327男人天堂| 久久机热/这里只有| 亚洲av一妻不如妾| 国产乱子伦一二三区| 午夜精品一区二区三区福利视频| 男人的天堂av日韩亚洲| 国产高清精品一区二区三区| 婷婷色国产黑丝少妇勾搭AV| 久久久久91精品推荐99| 99热久久极品热亚洲| 都市家庭人妻激情自拍视频| 日韩熟女av天堂系列| 91色网站免费在线观看| 日韩欧美在线观看不卡一区二区| 在线观看av观看av| 端庄人妻堕落挣扎沉沦| 欧美成人黄片一区二区三区 | 色婷婷精品大在线观看| 亚洲中文字字幕乱码| 亚洲一级av无码一级久久精品| 91天堂天天日天天操| 99热99这里精品6国产| 国产精品精品精品999| av在线观看网址av| 红杏久久av人妻一区| 18禁网站一区二区三区四区| 视频一区二区三区高清在线| 19一区二区三区在线播放| 国产精品系列在线观看一区二区 | 亚洲公开视频在线观看| 午夜毛片不卡免费观看视频| 国产又大又黄免费观看| 漂亮 人妻被中出中文| 大香蕉大香蕉在线有码 av| 强行扒开双腿猛烈进入免费版| 综合激情网激情五月天| 国产麻豆91在线视频| 韩国AV无码不卡在线播放| 国产亚洲成人免费在线观看| wwwxxx一级黄色片| wwwxxx一级黄色片| 一区二区三区另类在线| 国产女人被做到高潮免费视频 | 中文字幕日韩精品日本| 久久久久久久久久久免费女人| 日本一二三中文字幕| 国产黄色大片在线免费播放| 亚洲成人黄色一区二区三区| 极品粉嫩小泬白浆20p主播| 抽查舔水白紧大视频| sw137 中文字幕 在线| 18禁精品网站久久| 插逼视频双插洞国产操逼插洞| 久久久久只精品国产三级| 亚洲精品午夜久久久久| 色呦呦视频在线观看视频| 午夜福利资源综合激情午夜福利资| 亚洲av午夜免费观看| 色花堂在线av中文字幕九九 | 在线国产中文字幕视频| 亚洲在线免费h观看网站| 成年午夜影片国产片| 1区2区3区不卡视频| 在线播放一区二区三区Av无码| 亚洲av自拍偷拍综合| 人妻另类专区欧美制服| 视频久久久久久久人妻| 五十路熟女人妻一区二区9933| 清纯美女在线观看国产| 久久久制服丝袜中文字幕| 精品一区二区三区三区色爱| 国产1区,2区,3区| 日韩精品电影亚洲一区| 久草视频在线一区二区三区资源站| 中文 成人 在线 视频| 国产亚洲成人免费在线观看 | 国产精品人妻一区二区三区网站| 国产精品手机在线看片| 国产一区av澳门在线观看| 天天日天天敢天天干| 内射久久久久综合网| 国产在线观看黄色视频| 欧美一级视频一区二区| 91在线视频在线精品3| 国产成人自拍视频播放| 亚洲免费av在线视频| 超pen在线观看视频公开97| 激情五月婷婷免费视频| 女同久久精品秋霞网| 97资源人妻免费在线视频| 999久久久久999| 国产使劲操在线播放| 亚洲国产40页第21页| 含骚鸡巴玩逼逼视频| 黄片三级三级三级在线观看| 午夜精彩视频免费一区| 日日夜夜大香蕉伊人| 亚洲成人国产综合一区| 经典国语激情内射视频| 五月精品丁香久久久久福利社| 成人av久久精品一区二区| 日韩国产乱码中文字幕| 国产91嫩草久久成人在线视频| 国产亚洲欧美视频网站| 最新91精品视频在线| 18禁无翼鸟成人在线| 97人妻色免费视频| 日本欧美视频在线观看三区| 在线视频自拍第三页| 99精品视频在线观看免费播放 | 国产剧情演绎系列丝袜高跟| 日本一区精品视频在线观看| 久久热这里这里只有精品| 人妻熟女在线一区二区| 天天操天天爽天天干| 婷婷激情四射在线观看视频| 欧美麻豆av在线播放| 中英文字幕av一区| 亚洲av无女神免非久久| 伊人成人在线综合网| 超碰公开大香蕉97| 少妇人妻100系列| 日韩精品啪啪视频一道免费| 日韩美av高清在线| 亚洲变态另类色图天堂网| 韩国黄色一级二级三级| 亚洲在线一区二区欧美| 经典国语激情内射视频| 中文字幕视频一区二区在线观看| 女生被男生插的视频网站| 国产亚洲天堂天天一区| 男生舔女生逼逼视频| 日本高清在线不卡一区二区| 日本午夜爽爽爽爽爽视频在线观看| 欧美精品激情在线最新观看视频| 粉嫩av懂色av蜜臀av| 超污视频在线观看污污污| 少妇被强干到高潮视频在线观看| 淫秽激情视频免费观看| 亚洲综合乱码一区二区| 男生用鸡操女生视频动漫 | 国产视频一区二区午夜| 91麻豆精品传媒国产黄色片| 高清成人av一区三区| 久久精品国产亚洲精品166m| 日本女大学生的黄色小视频| 成人免费公开视频无毒| 欧美精品国产综合久久| 韩国三级aaaaa高清视频| 视频一区 视频二区 视频| 天天通天天透天天插| 国产aⅴ一线在线观看| 日韩欧美亚洲熟女人妻| 美女 午夜 在线视频| 91精品激情五月婷婷在线| 国产片免费观看在线观看| 动漫黑丝美女的鸡巴| 久久久超爽一二三av| 亚洲午夜高清在线观看| 天堂av在线官网中文| 一区二区三区日韩久久| 日本一区精品视频在线观看| 99亚洲美女一区二区三区| 99人妻视频免费在线| 家庭女教师中文字幕在线播放| 亚洲综合图片20p| 一区二区三区综合视频| 97精品综合久久在线| 亚洲综合另类精品小说| 精品一区二区三四区| 啊慢点鸡巴太大了啊舒服视频| 欧美久久久久久三级网| 40道精品招牌菜特色| 曰本无码人妻丰满熟妇啪啪| 一区二区在线观看少妇| 国产精品国色综合久久| 中文字幕第三十八页久久| 在线成人日韩av电影| 在线网站你懂得老司机| 高潮视频在线快速观看国家快速| 天天干天天操天天玩天天射| 一本一本久久a久久精品综合不卡| 国产麻豆精品人妻av| 538精品在线观看视频| 老司机午夜精品视频资源| 色伦色伦777国产精品| 91麻豆精品秘密入口在线观看| 日本性感美女三级视频| 少妇深喉口爆吞精韩国| 社区自拍揄拍尻屁你懂的| 亚洲精品乱码久久久久久密桃明 | 丁香花免费在线观看中文字幕| 国产V亚洲V天堂无码欠欠| 午夜美女福利小视频| 国内资源最丰富的网站| 日韩欧美亚洲熟女人妻| 午夜av一区二区三区| 国产精品三级三级三级| sw137 中文字幕 在线| 高清成人av一区三区| 免费在线看的黄片视频| 人妻少妇亚洲精品中文字幕| 中文字幕人妻一区二区视频| 亚洲少妇高潮免费观看| 一色桃子久久精品亚洲| 久久一区二区三区人妻欧美| 成人av在线资源网站| 偷拍美女一区二区三区| 91av精品视频在线| 亚洲综合在线视频可播放| 福利午夜视频在线合集| 香港一级特黄大片在线播放| 亚洲精品欧美日韩在线播放| 黄色成年网站午夜在线观看 | 一个色综合男人天堂| 日本少妇人妻xxxxx18| 天美传媒mv视频在线观看| www日韩毛片av| 激情色图一区二区三区| 日日夜夜大香蕉伊人| 天天操夜夜骑日日摸| 一二三中文乱码亚洲乱码one| 天天操,天天干,天天射| 91p0rny九色露脸熟女| 97人妻夜夜爽二区欧美极品| 国产91精品拍在线观看| 不卡一区一区三区在线| 57pao国产一区二区| 日日夜夜大香蕉伊人| 偷拍3456eee| 和邻居少妇愉情中文字幕| 黑人乱偷人妻中文字幕| 97精品视频在线观看| 久草极品美女视频在线观看| 亚洲一区av中文字幕在线观看| 亚洲精品久久视频婷婷| 亚洲激情唯美亚洲激情图片| 一级黄片大鸡巴插入美女| xxx日本hd高清| 18禁美女黄网站色大片下载| 欧美va亚洲va天堂va| 97小视频人妻一区二区| 中文字幕,亚洲人妻| 亚洲午夜在线视频福利| 午夜毛片不卡免费观看视频| 93精品视频在线观看| 亚洲 欧美 自拍 偷拍 在线| 97超碰人人搞人人| 亚国产成人精品久久久| 国际av大片在线免费观看| 扒开腿挺进肉嫩小18禁视频| a v欧美一区=区三区| 最新91精品视频在线| 成人资源在线观看免费官网| 偷拍自拍亚洲美腿丝袜| 日韩伦理短片在线观看| 人妻丝袜精品中文字幕| 国产成人精品午夜福利训2021| 中国黄色av一级片| 一个人免费在线观看ww视频| 91人妻人人做人人爽在线| 日韩美女搞黄视频免费| 99国内小视频在现欢看| 国产一线二线三线的区别在哪| 色综合久久久久久久久中文| 巨乳人妻日下部加奈被邻居中出| 宅男噜噜噜666国产| 操日韩美女视频在线免费看| 亚洲嫩模一区二区三区| 天天日天天天天天天天天天天 | 97国产在线观看高清| 蜜桃视频在线欧美一区| 天天艹天天干天天操| 2022天天干天天操| 天天干天天操天天摸天天射| 日韩av中文在线免费观看| 一二三中文乱码亚洲乱码one | 岛国毛片视频免费在线观看| 国产中文精品在线观看| 美女小视频网站在线| 国产乱子伦一二三区| 天天插天天色天天日| 一二三区在线观看视频| 亚洲av日韩av网站| 福利视频网久久91| 色婷婷六月亚洲综合香蕉| nagger可以指黑人吗| 中文字幕+中文字幕| av无限看熟女人妻另类av| 免费十精品十国产网站| 人妻少妇性色欲欧美日韩| sw137 中文字幕 在线| 18禁网站一区二区三区四区| 国产精品熟女久久久久浪潮| 亚洲的电影一区二区三区| 国产日韩av一区二区在线| 亚洲 国产 成人 在线| 国产97视频在线精品| 色哟哟在线网站入口| 亚洲av无硬久久精品蜜桃| 亚洲丝袜老师诱惑在线观看| 啊啊啊视频试看人妻| 亚洲第一伊人天堂网| 亚洲美女高潮喷浆视频| 91精品国产麻豆国产| 亚洲精品精品国产综合| 伊人综合aⅴ在线网| 欧美日本在线视频一区| 精品高潮呻吟久久av| 涩涩的视频在线观看视频| 制服丝袜在线人妻中文字幕| 亚洲中文字字幕乱码| 青青青青青操视频在线观看| 淫秽激情视频免费观看| 丝袜美腿欧美另类 中文字幕| 亚洲综合一区成人在线| 国产精品久久久黄网站| 国产视频在线视频播放| 日本一区美女福利视频| 爆乳骚货内射骚货内射在线| 亚洲国产成人最新资源| 首之国产AV医生和护士小芳| 97a片免费在线观看| av在线免费资源站| 蜜臀av久久久久久久| sejizz在线视频| 综合一区二区三区蜜臀| 亚洲国产中文字幕啊啊啊不行了| 欧美另类重口味极品在线观看| 中字幕人妻熟女人妻a62v网 | 国产成人一区二区三区电影网站| 视频一区二区在线免费播放| 国产激情av网站在线观看| 日韩北条麻妃一区在线| 欧美亚洲一二三区蜜臀| av男人天堂狠狠干| 自拍偷拍 国产资源| 亚洲成人免费看电影| 人妻少妇中文有码精品| 福利国产视频在线观看| 中文字幕日韩91人妻在线| 538精品在线观看视频| 大陆精品一区二区三区久久| 欧美一区二区三区乱码在线播放 | 2025年人妻中文字幕乱码在线 | 性感美女诱惑福利视频| 中文字幕在线一区精品| 不卡一区一区三区在线| 亚洲在线一区二区欧美| 97国产福利小视频合集| 国产1区,2区,3区| 成年午夜影片国产片| 毛茸茸的大外阴中国视频| 2022天天干天天操| 五月精品丁香久久久久福利社| 成人伊人精品色xxxx视频| 天天日天天摸天天爱| 日韩中文字幕在线播放第二页| 中文字幕无码一区二区免费| 最新国产亚洲精品中文在线| 亚洲av无硬久久精品蜜桃| 亚洲精品中文字幕下载| 91精品国产黑色丝袜| 狍和女人的王色毛片| 亚洲一区二区三区精品视频在线| 久久久久五月天丁香社区| 国产在线一区二区三区麻酥酥| 男人天堂色男人av| 亚洲一区二区三区久久受| 啊用力插好舒服视频| 婷婷色国产黑丝少妇勾搭AV | 国产在线自在拍91国语自产精品| 国产成人综合一区2区| 久久机热/这里只有| 手机看片福利盒子日韩在线播放| 偷拍自拍亚洲视频在线观看| 福利午夜视频在线观看| 亚洲免费av在线视频| 中文字幕在线欧美精品| 天天干天天日天天干天天操| 亚洲码av无色中文| 漂亮 人妻被中出中文| 欧美日本国产自视大全| 日本一道二三区视频久久| 欧美亚洲免费视频观看| 亚洲 欧美 自拍 偷拍 在线| 大香蕉日本伊人中文在线| 久久永久免费精品人妻专区| 国产精品久久久黄网站| 极品性荡少妇一区二区色欲| 粉嫩av懂色av蜜臀av| 啊啊啊想要被插进去视频| 中文亚洲欧美日韩无线码| 欧美一区二区三区高清不卡tv | 日韩av免费观看一区| 天天干狠狠干天天操| 青青青爽视频在线播放| 大骚逼91抽插出水视频| 色哟哟在线网站入口| caoporm超碰国产| 亚洲综合另类欧美久久| 久精品人妻一区二区三区| 国产精品自拍在线视频| 精品国产亚洲av一淫| 精品一区二区三区欧美| 91精品国产91久久自产久强| 中国熟女一区二区性xx| 人人妻人人爱人人草| 91精品国产综合久久久蜜| 亚洲国产成人在线一区| 免费男阳茎伸入女阳道视频 | 国产之丝袜脚在线一区二区三区| 欧美特色aaa大片| 欧美一级片免费在线成人观看| 91高清成人在线视频| 中文字幕最新久久久| 天天日天天鲁天天操| 日韩精品中文字幕在线| 99精品免费观看视频| 少妇ww搡性bbb91| 大肉大捧一进一出好爽在线视频| 91人妻精品一区二区在线看| 无码日韩人妻精品久久| 老师啊太大了啊啊啊尻视频| 国产精品欧美日韩区二区| 天天摸天天日天天操| 在线视频免费观看网| 4个黑人操素人视频网站精品91| 天天夜天天日天天日| 国产综合高清在线观看| 5528327男人天堂| 亚洲欧美自拍另类图片| 2020av天堂网在线观看| 日本av在线一区二区三区| 亚洲伊人久久精品影院一美女洗澡 | 欧美日本在线观看一区二区| 美女福利写真在线观看视频| 亚洲av香蕉一区区二区三区犇| 国产精品成人xxxx| 大胸性感美女羞爽操逼毛片| 中国无遮挡白丝袜二区精品| 美洲精品一二三产区区别| 国产chinesehd精品麻豆| 福利在线视频网址导航| 亚洲的电影一区二区三区| 日本真人性生活视频免费看| 国产中文精品在线观看| 日韩a级精品一区二区| 在线观看av观看av| 特级无码毛片免费视频播放| av手机在线观播放网站| 青青青青青青青青青青草青青| 人妻3p真实偷拍一二区| 在线观看av观看av| 中文字幕一区的人妻欧美日韩| 77久久久久国产精产品| 91九色porny蝌蚪国产成人| 日本www中文字幕| 最新激情中文字幕视频| 天天日天天添天天爽| 欧美viboss性丰满| 一区二区三区四区视频| 国产精品一区二区av国| 国产丰满熟女成人视频| 天天操天天污天天射| 一色桃子久久精品亚洲| 又粗又硬又猛又爽又黄的| 91av精品视频在线| 亚洲高清一区二区三区视频在线| 成年人黄色片免费网站| 亚洲成av人无码不卡影片一| 97人妻夜夜爽二区欧美极品| 91社福利《在线观看| 欧美成一区二区三区四区| 亚洲自拍偷拍精品网| 欧美精品黑人性xxxx| 亚洲精品成人网久久久久久小说| 日韩二区视频一线天婷婷五| 亚洲精品午夜aaa久久| 男人天堂最新地址av| 欧美亚洲免费视频观看| 18禁污污污app下载| 亚洲高清国产一区二区三区| 精品视频中文字幕在线播放| 日本一二三中文字幕| 免费高清自慰一区二区三区网站 | gay gay男男瑟瑟在线网站| 国产精品黄片免费在线观看| av在线播放国产不卡| 免费成人va在线观看| 中出中文字幕在线观看| 久久久久久97三级| AV无码一区二区三区不卡| 国产九色91在线观看精品| 亚洲精品久久视频婷婷| 免费69视频在线看| 成年人免费看在线视频| 青青青青青青青在线播放视频| 少妇一区二区三区久久久| 免费岛国喷水视频在线观看| aiss午夜免费视频| 亚洲av自拍天堂网| 自拍偷拍日韩欧美亚洲| 青青尤物在线观看视频网站| 亚洲自拍偷拍综合色| 国产黄色片在线收看| 国产+亚洲+欧美+另类| 中文字幕亚洲中文字幕| 免费高清自慰一区二区三区网站 | 亚洲最大黄 嗯色 操 啊| 888欧美视频在线| 亚洲老熟妇日本老妇| 亚洲视频在线观看高清| 国产精品国产三级国产午| 国产女人被做到高潮免费视频| 亚洲天天干 夜夜操| 六月婷婷激情一区二区三区| 亚洲综合在线观看免费| 中文字幕日韩精品就在这里| 18禁精品网站久久| 久久永久免费精品人妻专区| 91超碰青青中文字幕| 最新国产精品网址在线观看| 日韩美av高清在线| 视频啪啪啪免费观看| 亚洲综合色在线免费观看| 国产aⅴ一线在线观看| 国产综合视频在线看片| 亚洲最大黄了色网站| 中文字幕高清免费在线人妻| av无限看熟女人妻另类av| 青青青视频手机在线观看| 一区国内二区日韩三区欧美| 一级黄片久久久久久久久| 91免费观看国产免费| 蝴蝶伊人久久中文娱乐网| 97成人免费在线观看网站| 韩国AV无码不卡在线播放| 91精品综合久久久久3d动漫 | 91片黄在线观看喷潮| 在线观看av2025| av老司机精品在线观看| 亚洲色偷偷综合亚洲AV伊人| 黄色三级网站免费下载| 欧美一区二区三区啪啪同性| 青青色国产视频在线| 亚洲av黄色在线网站| 成人在线欧美日韩国产| 超污视频在线观看污污污| 经典亚洲伊人第一页| 久久久噜噜噜久久熟女av| 888亚洲欧美国产va在线播放| av黄色成人在线观看| 亚欧在线视频你懂的| 一区二区三区久久中文字幕| 国产欧美日韩第三页| 欧美交性又色又爽又黄麻豆| 国产日韩一区二区在线看| 77久久久久国产精产品| 亚洲国产欧美一区二区三区久久| 日韩一区二区电国产精品| 超碰97免费人妻麻豆| 晚上一个人看操B片| 2020中文字幕在线播放| 99国产精品窥熟女精品| 精品日产卡一卡二卡国色天香| 日本特级片中文字幕| 97色视频在线观看| 欧美黑人性猛交xxxxⅹooo| 97精品人妻一区二区三区精品| 大肉大捧一进一出好爽在线视频| 香蕉av影视在线观看| 韩国亚洲欧美超一级在线播放视频| 青青草成人福利电影| 欧美特色aaa大片| 在线观看操大逼视频| 久久尻中国美女视频| av日韩在线观看大全| 免费一级特黄特色大片在线观看| 亚洲国产精品美女在线观看| 国产性感美女福利视频| 在线播放 日韩 av| 亚洲嫩模一区二区三区| 毛茸茸的大外阴中国视频| 男人靠女人的逼视频| 偷拍自拍福利视频在线观看| 一区二区熟女人妻视频| 青青青视频手机在线观看| 日韩美在线观看视频黄| 新97超碰在线观看| 一区二区久久成人网| 青青在线视频性感少妇和隔壁黑丝| 成年午夜免费无码区| 18禁美女黄网站色大片下载| 成人高清在线观看视频| 天堂v男人视频在线观看| 综合国产成人在线观看| 91www一区二区三区| 91人妻人人做人人爽在线| 啪啪啪啪啪啪啪啪av| 夜夜操,天天操,狠狠操| 日韩少妇人妻精品无码专区| 麻豆精品成人免费视频| 女同久久精品秋霞网| 黄色av网站免费在线| 一区二区三区日本伦理| 国产精品手机在线看片| 亚洲欧美人精品高清| 亚洲卡1卡2卡三卡四老狼| 日本男女操逼视频免费看| 男人靠女人的逼视频| 亚洲综合乱码一区二区| 精品欧美一区二区vr在线观看 | 国产在线免费观看成人| av手机在线免费观看日韩av| 天天日夜夜操天天摸| 55夜色66夜色国产精品站| 日韩激情文学在线视频| 视频在线免费观看你懂得| 老鸭窝在线观看一区| 国产精品人久久久久久| 欧美视频综合第一页| www日韩a级s片av| 日本xx片在线观看| 熟女视频一区,二区,三区| 99国内小视频在现欢看| 日韩欧美在线观看不卡一区二区| 白白操白白色在线免费视频 | 欧美黄色录像免费看的| caoporn蜜桃视频| 特级欧美插插插插插bbbbb| sejizz在线视频| 5528327男人天堂| 天天干狠狠干天天操| 午夜福利资源综合激情午夜福利资| 日本午夜福利免费视频| 亚洲av自拍偷拍综合| 乱亲女秽乱长久久久| 大肉大捧一进一出好爽在线视频| 男人的天堂在线黄色| 久久久久久cao我的性感人妻| 天天日天天舔天天射进去| 天天操天天插天天色| 免费国产性生活视频| 国产视频网站一区二区三区| 日本韩国亚洲综合日韩欧美国产| 日本少妇在线视频大香蕉在线观看| 自拍偷拍亚洲欧美在线视频| 亚洲欧美清纯唯美另类| 久久精品36亚洲精品束缚| 97人人模人人爽人人喊| 日韩人妻xxxxx| av在线免费观看亚洲天堂| 亚洲福利天堂久久久久久 | 欧美成人黄片一区二区三区| 青青青青操在线观看免费| 韩国亚洲欧美超一级在线播放视频 | 超碰在线中文字幕一区二区| 日本在线一区二区不卡视频| 亚洲国产精品免费在线观看| 日本最新一二三区不卡在线| av俺也去在线播放| 日本韩国亚洲综合日韩欧美国产| 淫秽激情视频免费观看| 日本少妇的秘密免费视频| 一区二区三区av高清免费| 亚洲精品亚洲人成在线导航| 果冻传媒av一区二区三区| 99热久久极品热亚洲| 夜夜骑夜夜操夜夜奸| 天天干天天操天天爽天天摸| 蜜桃色婷婷久久久福利在线| 亚洲国产欧美一区二区三区久久| 欧美3p在线观看一区二区三区| 午夜大尺度无码福利视频| 日本人妻少妇18—xx| 国产性生活中老年人视频网站| 精彩视频99免费在线| 国产品国产三级国产普通话三级| 欧美成人一二三在线网| 自拍偷拍亚洲精品第2页| 一区二区三区另类在线| 午夜青青草原网在线观看| 午夜精品一区二区三区更新| okirakuhuhu在线观看| 欧美3p在线观看一区二区三区| 免费观看国产综合视频| 国产综合高清在线观看| 国产高清精品极品美女| 日本美女成人在线视频| 欧美性感尤物人妻在线免费看| 中文字幕在线观看国产片| 大白屁股精品视频国产| 99精品亚洲av无码国产另类| 伊人综合免费在线视频| 欧美一区二区三区激情啪啪啪| 风流唐伯虎电视剧在线观看 | 888欧美视频在线| 色爱av一区二区三区| av老司机精品在线观看| 国产美女精品福利在线| 久久久久久久久久一区二区三区| 成人区人妻精品一区二视频| 超污视频在线观看污污污| 91 亚洲视频在线观看| 久青青草视频手机在线免费观看| 日韩黄色片在线观看网站| 日韩视频一区二区免费观看| 色哟哟在线网站入口| 92福利视频午夜1000看 | 国产日韩欧美美利坚蜜臀懂色| 日韩精品一区二区三区在线播放| 日韩精品激情在线观看| 亚洲一区二区三区uij| 在线成人日韩av电影| 久久99久久99精品影院| 久久久精品999精品日本| 久久这里有免费精品| 亚洲国际青青操综合网站 | 欧美日本在线视频一区| 久久久超爽一二三av| 亚洲国产中文字幕啊啊啊不行了| 97香蕉碰碰人妻国产樱花| 中文字幕 亚洲av| 国产精品欧美日韩区二区| 美女被肏内射视频网站| 久久综合老鸭窝色综合久久 | 亚洲中文字幕乱码区| 人妻激情图片视频小说| 一色桃子人妻一区二区三区| 天天操天天弄天天射| 99re久久这里都是精品视频| 91av中文视频在线| 绯色av蜜臀vs少妇| 青青青青青青草国产| 亚洲国产中文字幕啊啊啊不行了| 国产成人午夜精品福利| 午夜精品在线视频一区| 亚洲视频在线视频看视频在线| 丝袜肉丝一区二区三区四区在线 | 久精品人妻一区二区三区| 国产精品视频一区在线播放| 天天操,天天干,天天射| 一区二区三区四区视频| 五十路av熟女松本翔子| 国产精品久久9999| 天天操天天干天天日狠狠插 | 五十路老熟女码av| 欧美特色aaa大片| 亚洲国产欧美一区二区丝袜黑人| 日日日日日日日日夜夜夜夜夜夜| 亚洲av黄色在线网站| 国产又粗又猛又爽又黄的视频美国| 久久农村老妇乱69系列| 久久综合老鸭窝色综合久久| 青青青艹视频在线观看| 国产中文精品在线观看| 成人性黑人一级av| 久久www免费人成一看片| 亚洲成人情色电影在线观看| 青青尤物在线观看视频网站| 国产综合精品久久久久蜜臀| 国产精品久久综合久久| 999热精品视频在线| 日韩美女精品视频在线观看网站| 国产免费av一区二区凹凸四季| 大屁股肉感人妻中文字幕在线| 亚洲免费国产在线日韩| 午夜福利人人妻人人澡人人爽| 亚洲国产精品中文字幕网站| 成人福利视频免费在线| 都市家庭人妻激情自拍视频| 日本a级视频老女人| 国产91久久精品一区二区字幕| 新婚人妻聚会被中出| 扒开腿挺进肉嫩小18禁视频| 日韩一区二区电国产精品| 国产av一区2区3区| jiuse91九色视频| 大胆亚洲av日韩av| 最新中文字幕乱码在线| 97精品综合久久在线| 91久久人澡人人添人人爽乱| 日美女屁股黄邑视频| 香蕉片在线观看av| 国产精品大陆在线2019不卡| 天天日天天干天天爱| 成人24小时免费视频| 国产成人精品一区在线观看| 免费大片在线观看视频网站| 白白操白白色在线免费视频 | 精品国产午夜视频一区二区| 午夜美女少妇福利视频| 91精品国产91久久自产久强| 中文字幕高清资源站| 国产三级片久久久久久久| 成熟熟女国产精品一区| 成人精品在线观看视频| 玖玖一区二区在线观看| 青青青青爽手机在线| 青青社区2国产视频| 91麻豆精品久久久久| 亚洲人妻30pwc| 亚洲高清国产拍青青草原| 91试看福利一分钟| 中文字幕高清资源站| 成人av天堂丝袜在线观看| 天天操天天爽天天干| 亚洲熟女久久久36d| 欧美日韩v中文在线| 老司机在线精品福利视频| 国产超码片内射在线| 国产午夜亚洲精品麻豆| 红杏久久av人妻一区| 国产乱子伦精品视频潮优女| 亚洲精品欧美日韩在线播放| 欧美日韩情色在线观看| 国产av自拍偷拍盛宴| 欧美日韩一区二区电影在线观看 | 91中文字幕最新合集| 亚洲一区二区三区精品视频在线| 91精品免费久久久久久| 免费成人av中文字幕| 欧美男同性恋69视频| 福利午夜视频在线合集| 亚洲欧美综合在线探花| 粉嫩av蜜乳av蜜臀| 视频 一区二区在线观看| 亚洲免费福利一区二区三区| 狠狠操狠狠操免费视频| 亚洲精品乱码久久久久久密桃明| 国产精品视频欧美一区二区| 欧美亚洲国产成人免费在线| 亚洲av男人的天堂你懂的| 免费看美女脱光衣服的视频| 久草视频 久草视频2| 午夜精品久久久久久99热| 很黄很污很色的午夜网站在线观看| 日韩欧美国产精品91| 国产又粗又硬又猛的毛片视频 | 在线视频免费观看网| 亚洲激情偷拍一区二区| 偷拍自拍国产在线视频| 91免费观看国产免费| 亚洲最大黄了色网站| 欧美aa一级一区三区四区| 韩国一级特黄大片做受| 天天综合天天综合天天网| 国内精品在线播放第一页| 亚洲嫩模一区二区三区| 福利在线视频网址导航| 久久久久久久久久久久久97| 97色视频在线观看| 亚洲精品久久综合久| 亚洲综合另类精品小说| 女生自摸在线观看一区二区三区 | 日韩欧美国产精品91| 天天日天天干天天舔天天射| 99精品免费观看视频| 55夜色66夜色国产精品站| 国产精品一二三不卡带免费视频| 亚洲一区二区三区在线高清| 在线观看一区二区三级| 中文人妻AV久久人妻水| 亚洲男人让女人爽的视频| 91成人精品亚洲国产| 亚洲av人人澡人人爽人人爱| 中文字幕视频一区二区在线观看| 亚洲乱码中文字幕在线| 日本av熟女在线视频| 狍和女人的王色毛片| 丝袜长腿第一页在线| 亚洲国产在人线放午夜| 和邻居少妇愉情中文字幕| 黄色录像鸡巴插进去| 岛国av高清在线成人在线| 涩涩的视频在线观看视频| 亚洲成人av一区久久| 绝顶痉挛大潮喷高潮无码| 熟妇一区二区三区高清版| 天天日天天日天天射天天干| 免费男阳茎伸入女阳道视频| 女警官打开双腿沦为性奴| 操操网操操伊剧情片中文字幕网| 小穴多水久久精品免费看| 久久麻豆亚洲精品av| 日韩欧美中文国产在线| 久草视频福利在线首页| 国产综合视频在线看片| 一区二区在线观看少妇| 91免费黄片可看视频 | 亚洲高清免费在线观看视频| 亚洲精品国产综合久久久久久久久| 婷婷六月天中文字幕| 中文字幕第1页av一天堂网 | 亚洲图库另类图片区| 亚洲精品一区二区三区老狼| 日韩中文字幕福利av| 日本女人一级免费片| 午夜精品久久久久麻豆影视| 夜夜嗨av蜜臀av| 国内精品在线播放第一页| 日韩在线中文字幕色| 亚洲另类伦春色综合小| 日韩美av高清在线| 在线免费91激情四射| 精品久久久久久久久久久99| 欧美专区日韩专区国产专区| 色狠狠av线不卡香蕉一区二区| 欧美色呦呦最新网址| 夏目彩春在线中文字幕| 精品少妇一二三视频在线| 国产片免费观看在线观看| 78色精品一区二区三区| 综合国产成人在线观看| 激情五月婷婷综合色啪| 夜色17s精品人妻熟女| 少妇深喉口爆吞精韩国| 98精产国品一二三产区区别| 91免费观看国产免费| 2025年人妻中文字幕乱码在线| 国产夫妻视频在线观看免费 | 91精品激情五月婷婷在线| 一区二区三区另类在线| 天天干天天操天天扣| 亚洲伊人av天堂有码在线| 成人久久精品一区二区三区| 亚洲福利精品福利精品福利| www久久久久久久久久久| 98视频精品在线观看| 岛国黄色大片在线观看| 久久久久久97三级| 不卡一不卡二不卡三| 五十路丰满人妻熟妇| 中文字幕av男人天堂| 久草免费人妻视频在线| 亚洲另类伦春色综合小| 91大神福利视频网| 亚洲天堂第一页中文字幕| 亚洲女人的天堂av| 亚洲天堂有码中文字幕视频| 国产又粗又猛又爽又黄的视频美国| 中文字幕无码一区二区免费| 全国亚洲男人的天堂| 美日韩在线视频免费看| 国产午夜无码福利在线看| 美女 午夜 在线视频| 亚洲精品成人网久久久久久小说 | 超污视频在线观看污污污| 成人免费毛片aaaa| 国产精品自拍视频大全| 午夜dv内射一区区| 成人av亚洲一区二区| 免费观看理论片完整版| 青青色国产视频在线| 玖玖一区二区在线观看| 久久这里有免费精品| 日本女大学生的黄色小视频| 欧美视频一区免费在线| 亚洲1卡2卡三卡4卡在线观看| 无套猛戳丰满少妇人妻| 一区二区三区激情在线| 亚洲欧美色一区二区| 91麻豆精品久久久久| lutube在线成人免费看| 超碰在线中文字幕一区二区| 在线免费91激情四射 | 日本少妇精品免费视频| 中文字幕免费在线免费| 99亚洲美女一区二区三区| 日本高清撒尿pissing| 亚洲天天干 夜夜操| 亚洲狠狠婷婷综合久久app| 91中文字幕最新合集| 三级av中文字幕在线观看| 亚洲午夜福利中文乱码字幕| 国产精品一区二区av国| 2021久久免费视频| av视网站在线观看| 自拍偷拍亚洲欧美在线视频| 精品一区二区三区三区色爱| 日韩a级黄色小视频| 在线国产日韩欧美视频| 91综合久久亚洲综合| 精品视频一区二区三区四区五区| 后入美女人妻高清在线| 激情国产小视频在线| 黄片大全在线观看观看| 午夜dv内射一区区| jiujiure精品视频在线| 天堂v男人视频在线观看| 男人的天堂一区二区在线观看| 天天射,天天操,天天说| 久久三久久三久久三久久| 久久久久国产成人精品亚洲午夜| 精品视频一区二区三区四区五区 | 亚洲最大免费在线观看| 在线免费观看黄页视频| 97少妇精品在线观看| 青青青激情在线观看视频| 在线观看欧美黄片一区二区三区 | 狠狠躁夜夜躁人人爽天天天天97| chinese国产盗摄一区二区 | 在线观看911精品国产| 一区二区三区的久久的蜜桃的视频 | 岳太深了紧紧的中文字幕| 欧美另类重口味极品在线观看| 亚洲成人精品女人久久久| 国产又大又黄免费观看| 人妻在线精品录音叫床| 亚洲久久午夜av一区二区| 日韩成人综艺在线播放| 亚洲伊人色一综合网| 在线免费观看黄页视频| 免费啪啪啪在线观看视频| 瑟瑟视频在线观看免费视频| 鸡巴操逼一级黄色气| 少妇人妻久久久久视频黄片| 骚货自慰被发现爆操| 日日操夜夜撸天天干| 国产精品久久久久久久精品视频 | 激情啪啪啪啪一区二区三区 | 日本人妻少妇18—xx| 非洲黑人一级特黄片| 亚洲欧美在线视频第一页| 久久久久久久亚洲午夜综合福利 | 蜜桃精品久久久一区二区| 亚洲高清国产一区二区三区| 美女小视频网站在线| 神马午夜在线观看视频| 日本a级视频老女人| 在线观看的a站 最新| 四川乱子伦视频国产vip| 97精品视频在线观看| 在线免费观看日本伦理| 国产午夜男女爽爽爽爽爽视频 | 99国产精品窥熟女精品| 午夜福利人人妻人人澡人人爽| 一二三中文乱码亚洲乱码one| 中文字幕一区二 区二三区四区| 久久这里只有精品热视频| 成年女人免费播放视频| 午夜精品一区二区三区城中村| 亚洲 欧美 精品 激情 偷拍| 18禁免费av网站| 换爱交换乱高清大片| 夜色17s精品人妻熟女| 99精品久久久久久久91蜜桃| 日本丰满熟妇大屁股久久| 一个色综合男人天堂| 中文字幕日韩精品日本| 天天通天天透天天插| yy6080国产在线视频| 久久久久久久一区二区三| 2012中文字幕在线高清| 国产在线自在拍91国语自产精品| 亚洲女人的天堂av| 鸡巴操逼一级黄色气| 精彩视频99免费在线| 美女福利视频导航网站| 黑人借宿ntr人妻的沦陷2| 只有精品亚洲视频在线观看| av欧美网站在线观看| 日本午夜福利免费视频| 亚洲综合自拍视频一区| 亚洲人一区二区中文字幕| 乱亲女秽乱长久久久| 欧美日韩激情啪啪啪| 国产精品3p和黑人大战| 国产精品自拍偷拍a| 在线制服丝袜中文字幕| 亚洲码av无色中文| 乱亲女秽乱长久久久| 欧美激情电影免费在线| 中文字幕第1页av一天堂网| 欧美激情精品在线观看| 中文字幕+中文字幕| 在线免费观看靠比视频的网站| 久久久噜噜噜久久熟女av| 亚洲精品麻豆免费在线观看| 亚洲成人情色电影在线观看| 中文字幕国产专区欧美激情| 欧美日韩人妻久久精品高清国产| 亚洲成人熟妇一区二区三区| 美女骚逼日出水来了| 中国视频一区二区三区| 天美传媒mv视频在线观看| 午夜精品在线视频一区| 一级a看免费观看网站| 国产麻豆乱子伦午夜视频观看| 国产视频一区二区午夜| 成人蜜桃美臀九一一区二区三区| 黑人进入丰满少妇视频| 国产亚洲精品视频合集| 亚洲av黄色在线网站| 不卡日韩av在线观看| 丝袜肉丝一区二区三区四区在线看| 亚洲最大免费在线观看| 91亚洲手机在线视频播放| 日本女人一级免费片| 日韩少妇人妻精品无码专区| 久久这里只有精品热视频| 国产精品一区二区三区蜜臀av| 日韩精品中文字幕播放| 亚洲成人线上免费视频观看| 亚洲无码一区在线影院| 青草青永久在线视频18| av乱码一区二区三区| 人妻少妇性色欲欧美日韩| 在线免费观看亚洲精品电影| 国产精品福利小视频a| 日本少妇人妻xxxxxhd| 91社福利《在线观看| 午夜毛片不卡免费观看视频| 黄色三级网站免费下载| 久久免看30视频口爆视频| 免费成人va在线观看| 久久久精品欧洲亚洲av| 国产日韩精品免费在线| 亚洲欧美清纯唯美另类| 国产精品成久久久久三级蜜臀av| 精品一区二区三区欧美| 91高清成人在线视频| 美女福利写真在线观看视频| ka0ri在线视频| 亚洲福利天堂久久久久久| 少妇被强干到高潮视频在线观看| 自拍偷拍vs一区二区三区| 最近中文2019年在线看| 男生用鸡操女生视频动漫| 91综合久久亚洲综合| 国产精品黄片免费在线观看| 高潮喷水在线视频观看| 激情综合治理六月婷婷| 亚洲国产免费av一区二区三区| 欧美一区二区三区乱码在线播放| 中文字幕无码日韩专区免费| 日本乱人一区二区三区| 色天天天天射天天舔| 日韩av熟妇在线观看| 亚洲 色图 偷拍 欧美| 久久永久免费精品人妻专区 | 天天夜天天日天天日| 欧美男同性恋69视频| 又黄又刺激的午夜小视频| 91av中文视频在线| 五十路息与子猛烈交尾视频| 国产成人自拍视频在线免费观看| 精品久久婷婷免费视频| 亚洲综合一区二区精品久久| av无限看熟女人妻另类av| 国产激情av网站在线观看| 精品一区二区三区三区色爱| 99精品视频在线观看婷婷| 91极品新人『兔兔』精品新作| av成人在线观看一区| 夫妻在线观看视频91| 人妻激情图片视频小说| 99久久成人日韩欧美精品| 久碰精品少妇中文字幕av| 成人av久久精品一区二区| 中文字幕人妻av在线观看| 人妻少妇亚洲一区二区| 男女啪啪视频免费在线观看| 国产精品欧美日韩区二区| 91免费福利网91麻豆国产精品 | 日韩无码国产精品强奸乱伦| 超级福利视频在线观看| 99精品久久久久久久91蜜桃| 五十路丰满人妻熟妇| 亚洲国产第一页在线观看| 黄色三级网站免费下载| 成人24小时免费视频| 无码日韩人妻精品久久| 插小穴高清无码中文字幕| 亚洲精品国产综合久久久久久久久| 人妻久久无码中文成人| 99国内小视频在现欢看| 熟女人妻在线中出观看完整版| 91麻豆精品秘密入口在线观看 | 欧美日韩一级黄片免费观看| 亚洲欧美成人综合在线观看| 免费大片在线观看视频网站| 精品欧美一区二区vr在线观看| 天天干天天操天天玩天天射| 夏目彩春在线中文字幕| 中文字幕在线永久免费播放| 51国产成人精品视频| 天天干天天爱天天色| 美女日逼视频免费观看| 另类av十亚洲av| 新97超碰在线观看| 在线免费观看黄页视频| 这里有精品成人国产99| 男女啪啪啪啪啪的网站| 亚洲欧美综合另类13p| av天堂中文字幕最新| 最新国产亚洲精品中文在线| 日本xx片在线观看| 在线免费观看国产精品黄色| 日韩少妇人妻精品无码专区| 亚洲 中文字幕在线 日韩| 在线观看视频污一区| 中文字幕日韩无敌亚洲精品| 色在线观看视频免费的| 五色婷婷综合狠狠爱| 宅男噜噜噜666免费观看| 18禁免费av网站| 九色porny九色9l自拍视频| 女同性ⅹxx女同hd| 国产一区av澳门在线观看| 人妻爱爱 中文字幕| 日韩av大胆在线观看| 美女在线观看日本亚洲一区| 高潮喷水在线视频观看| 国产大鸡巴大鸡巴操小骚逼小骚逼| 好吊操视频这里只有精品| 亚洲精品国品乱码久久久久| jul—619中文字幕在线| 天天射夜夜操狠狠干| 爆乳骚货内射骚货内射在线 | 中文字幕人妻被公上司喝醉在线| 91色秘乱一区二区三区| 中文字幕午夜免费福利视频| 人妻最新视频在线免费观看| 91久久精品色伊人6882| av中文字幕在线观看第三页| 亚洲欧美成人综合视频| 少妇露脸深喉口爆吞精| 亚洲精品福利网站图片| 亚洲区欧美区另类最新章节| 亚洲欧美国产综合777| 在线国产精品一区二区三区| 成人av在线资源网站| 久久三久久三久久三久久| 岛国黄色大片在线观看| 国产91久久精品一区二区字幕| 精品av国产一区二区三区四区| 欧美久久一区二区伊人| 午夜免费观看精品视频| 激情啪啪啪啪一区二区三区| av视网站在线观看| 精品久久久久久久久久久久人妻| 高潮视频在线快速观看国家快速| 亚洲av一妻不如妾| 亚洲另类综合一区小说| 视频一区二区综合精品| 亚洲欧美激情中文字幕| 日本高清在线不卡一区二区| 天天日天天摸天天爱| 红桃av成人在线观看| 无码中文字幕波多野不卡| 国产一区二区久久久裸臀| 国产精选一区在线播放| 国产黄网站在线观看播放| 天天操天天干天天日狠狠插| 欧美色呦呦最新网址| 91精品高清一区二区三区| 老司机99精品视频在线观看 | 人人妻人人澡人人爽人人dvl| 人人爱人人妻人人澡39| 男人的天堂一区二区在线观看| 欧美一区二区三区激情啪啪啪 | 新婚人妻聚会被中出| 青青青艹视频在线观看| 可以在线观看的av中文字幕| 国产亚洲视频在线二区| 久久精品在线观看一区二区| 天天躁夜夜躁日日躁a麻豆| 亚洲欧美激情国产综合久久久 | 亚洲高清视频在线不卡| 日本最新一二三区不卡在线| 五色婷婷综合狠狠爱| 天堂av在线最新版在线| 午夜久久久久久久99| 天天日夜夜干天天操| 欧美另类z0z变态| 伊人成人综合开心网| 性生活第二下硬不起来| 丝袜长腿第一页在线| 欧美在线一二三视频| 亚洲成人国产综合一区| 亚洲图片偷拍自拍区| 国产在线一区二区三区麻酥酥| 精品国产亚洲av一淫| a v欧美一区=区三区| 在线新三级黄伊人网| 亚洲国产在人线放午夜| 天干天天天色天天日天天射| 欧美精品一二三视频| 国产精品久久久久网| 欧美亚洲牲夜夜综合久久| 天天插天天色天天日| 精品美女久久久久久| 91精品国产91青青碰| 亚洲最大黄了色网站| 免费成人av中文字幕| 免费大片在线观看视频网站| av成人在线观看一区| 亚洲精品无码久久久久不卡| 黄网十四区丁香社区激情五月天 | 人妻丝袜榨强中文字幕| 亚洲午夜福利中文乱码字幕| 国产清纯美女al在线| 天天干天天日天天干天天操| 亚洲视频在线观看高清| 超碰公开大香蕉97| 一区二区三区国产精选在线播放| 国产露脸对白在线观看| 欧美偷拍亚洲一区二区| 日本韩国免费福利精品| 成人久久精品一区二区三区| 欧美亚洲国产成人免费在线| 亚洲第17页国产精品| 极品性荡少妇一区二区色欲| 大香蕉日本伊人中文在线| 国产真实乱子伦a视频| 粉嫩小穴流水视频在线观看| 国产a级毛久久久久精品| 亚洲精品乱码久久久本| 51国产成人精品视频| 亚洲变态另类色图天堂网| 黄色片黄色片wyaa| 欧亚日韩一区二区三区观看视频| 这里只有精品双飞在线播放| 日美女屁股黄邑视频| 黄色大片男人操女人逼| ka0ri在线视频| 青青青青爽手机在线| 馒头大胆亚洲一区二区| caoporn蜜桃视频| 中文乱理伦片在线观看| 久久久久久97三级| 欧美日本在线观看一区二区| av天堂中文字幕最新| 99av国产精品欲麻豆| 日韩熟女系列一区二区三区| 大香蕉伊人国产在线| 中国产一级黄片免费视频播放| 又粗又硬又猛又爽又黄的| 亚洲高清免费在线观看视频| av亚洲中文天堂字幕网| 狠狠操狠狠操免费视频| 青青青青青免费视频| 欧洲黄页网免费观看| sspd152中文字幕在线| 国产精品一区二区av国| 极品性荡少妇一区二区色欲| 熟女在线视频一区二区三区| 国产精品国产三级国产精东| 最新激情中文字幕视频| 精品视频国产在线观看| 国产精品视频一区在线播放| 国产密臀av一区二区三| 777奇米久久精品一区| 婷婷激情四射在线观看视频| 国产精品国产三级国产午| 国产午夜亚洲精品不卡在线观看| 青草久久视频在线观看| 久久国产精品精品美女| 亚洲综合另类欧美久久| 国产福利小视频二区| 中文字幕在线观看国产片| 日本美女成人在线视频| 最新激情中文字幕视频| 亚洲国产精品中文字幕网站| av网址国产在线观看| 性色蜜臀av一区二区三区| 精品久久婷婷免费视频| 在线观看视频一区麻豆| 久久精品国产亚洲精品166m| 中字幕人妻熟女人妻a62v网| 女同久久精品秋霞网| 男生舔女生逼逼的视频| 午夜精品一区二区三区4| 人人妻人人爱人人草| 国产乱弄免费视频观看| 午夜婷婷在线观看视频| a v欧美一区=区三区| 免费无码人妻日韩精品一区二区| 中国把吊插入阴蒂的视频| 亚洲综合另类欧美久久| 狠狠躁狠狠爱网站视频| 国产又大又黄免费观看| 91精品综合久久久久3d动漫| 中文字幕在线欧美精品| 不卡一不卡二不卡三| 美女视频福利免费看| 99精品国产aⅴ在线观看| 亚洲va天堂va国产va久| 久久精品久久精品亚洲人| lutube在线成人免费看| 春色激情网欧美成人| 2021年国产精品自拍| 93精品视频在线观看| 欧美一级片免费在线成人观看| 精品黑人一区二区三区久久国产| 国产自拍在线观看成人| 中文字幕综合一区二区| 日韩av有码中文字幕| 蝴蝶伊人久久中文娱乐网| 中文字幕熟女人妻久久久| 日韩欧美高清免费在线| 自拍偷拍,中文字幕| 亚洲2021av天堂| 欧美亚洲自偷自拍 在线| 特级无码毛片免费视频播放| 在线免费观看日本片| 免费一级特黄特色大片在线观看| 国产精品久久久久久久女人18| 2022天天干天天操| 国产剧情演绎系列丝袜高跟| 中文字幕网站你懂的| 新97超碰在线观看| 亚洲综合自拍视频一区| 果冻传媒av一区二区三区| 手机看片福利盒子日韩在线播放| 亚洲成人国产av在线| 久久久人妻一区二区| 91久久国产成人免费网站| 国产在线自在拍91国语自产精品 | 亚洲一区制服丝袜美腿| 欧美专区日韩专区国产专区| 成人30分钟免费视频| 亚洲2021av天堂| 国产欧美精品免费观看视频| 啊慢点鸡巴太大了啊舒服视频| 久久美欧人妻少妇一区二区三区| 一区二区三区日韩久久| 国产精品亚洲а∨天堂免| 免费人成黄页网站在线观看国产| 激情内射在线免费观看| 视频二区在线视频观看| 黄色大片男人操女人逼| 亚洲一区二区久久久人妻| 欧美黑人巨大性xxxxx猛交| 天堂中文字幕翔田av| 日本人妻精品久久久久久| 午夜影院在线观看视频羞羞羞| 国产视频一区在线观看| 亚洲一级特黄特黄黄色录像片| 亚洲va国产va欧美va在线| 日曰摸日日碰夜夜爽歪歪| 亚洲国产免费av一区二区三区| 在线免费视频 自拍| 久久精品久久精品亚洲人| 中文字幕一区二区亚洲一区| 日韩近亲视频在线观看| 在线新三级黄伊人网| 免费成人va在线观看| 成人区人妻精品一区二视频|