青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

Fork me on GitHub
隨筆 - 215  文章 - 13  trackbacks - 0
<2017年1月>
25262728293031
1234567
891011121314
15161718192021
22232425262728
2930311234


專注即時通訊及網(wǎng)游服務(wù)端編程
------------------------------------
Openresty 官方模塊
Openresty 標準模塊(Opm)
Openresty 三方模塊
------------------------------------
本博收藏大部分文章為轉(zhuǎn)載,并在文章開頭給出了原文出處,如有再轉(zhuǎn),敬請保留相關(guān)信息,這是大家對原創(chuàng)作者勞動成果的自覺尊重!!如為您帶來不便,請于本博下留言,謝謝配合。

常用鏈接

留言簿(1)

隨筆分類

隨筆檔案

相冊

Awesome

Blog

Book

GitHub

Link

搜索

  •  

積分與排名

  • 積分 - 220951
  • 排名 - 117

最新評論

閱讀排行榜

PHP高級編程之守護進程

http://netkiller.github.io/journal/php.daemon.html

Mr. Neo Chen (陳景峯), netkiller, BG7NYT

1. 什么是守護進程

守護進程是脫離于終端并且在后臺運行的進程。守護進程脫離于終端是為了避免進程在執(zhí)行過程中的信息在任何終端上顯示并且進程也不會被任何終端所產(chǎn)生的終端信息所打斷。

例如 apache, nginx, mysql 都是守護進程

2. 為什么開發(fā)守護進程

很多程序以服務(wù)形式存在,他沒有終端或UI交互,它可能采用其他方式與其他程序交互,如TCP/UDP Socket, UNIX Socket, fifo。程序一旦啟動便進入后臺,直到滿足條件他便開始處理任務(wù)。

3. 何時采用守護進程開發(fā)應(yīng)用程序

以我當前的需求為例,我需要運行一個程序,然后監(jiān)聽某端口,持續(xù)接受服務(wù)端發(fā)起的數(shù)據(jù),然后對數(shù)據(jù)分析處理,再將結(jié)果寫入到數(shù)據(jù)庫中; 我采用ZeroMQ實現(xiàn)數(shù)據(jù)收發(fā)。

如果我不采用守護進程方式開發(fā)該程序,程序一旦運行就會占用當前終端窗框,還有受到當前終端鍵盤輸入影響,有可能程序誤退出。

4. 守護進程的安全問題

我們希望程序在非超級用戶運行,這樣一旦由于程序出現(xiàn)漏洞被駭客控制,攻擊者只能繼承運行權(quán)限,而無法獲得超級用戶權(quán)限。

我們希望程序只能運行一個實例,不運行同事開啟兩個以上的程序,因為會出現(xiàn)端口沖突等等問題。

5. 怎樣開發(fā)守護進程

例 1. 多線程守護進程例示
			
<?php
class ExampleWorker extends Worker {

	#public function __construct(Logging $logger) {
	#	$this->logger = $logger;
	#}

	#protected $logger;
	protected  static $dbh;
	public function __construct() {

	}
	public function run(){
		$dbhost = '192.168.2.1';			// 數(shù)據(jù)庫服務(wù)器
		$dbport = 3306;
	    $dbuser = 'www';        			// 數(shù)據(jù)庫用戶名
        $dbpass = 'qwer123';             	// 數(shù)據(jù)庫密碼
		$dbname = 'example';				// 數(shù)據(jù)庫名

		self::$dbh  = new PDO("mysql:host=$dbhost;port=$dbport;dbname=$dbname", $dbuser, $dbpass, array(
			/* PDO::MYSQL_ATTR_INIT_COMMAND => 'SET NAMES \'UTF8\'', */
			PDO::MYSQL_ATTR_COMPRESS => true,
			PDO::ATTR_PERSISTENT => true
			)
		);

	}
	protected function getInstance(){
        return self::$dbh;
    }

}

/* the collectable class implements machinery for Pool::collect */
class Fee extends Stackable {
	public function __construct($msg) {
		$trades = explode(",", $msg);
		$this->data = $trades;
		print_r($trades);
	}

	public function run() {
		#$this->worker->logger->log("%s executing in Thread #%lu", __CLASS__, $this->worker->getThreadId() );

		try {
			$dbh  = $this->worker->getInstance();
			
			$insert = "INSERT INTO fee(ticket, login, volume, `status`) VALUES(:ticket, :login, :volume,'N')";
			$sth = $dbh->prepare($insert);
			$sth->bindValue(':ticket', $this->data[0]);
			$sth->bindValue(':login', $this->data[1]);
			$sth->bindValue(':volume', $this->data[2]);
			$sth->execute();
			$sth = null;
			
			/* ...... */
			
			$update = "UPDATE fee SET `status` = 'Y' WHERE ticket = :ticket and `status` = 'N'";
			$sth = $dbh->prepare($update);
			$sth->bindValue(':ticket', $this->data[0]);
			$sth->execute();
			//echo $sth->queryString;
			//$dbh = null;
		}
		catch(PDOException $e) {
			$error = sprintf("%s,%s\n", $mobile, $id );
			file_put_contents("mobile_error.log", $error, FILE_APPEND);
		}
	}
}

class Example {
	/* config */
	const LISTEN = "tcp://192.168.2.15:5555";
	const MAXCONN = 100;
	const pidfile = __CLASS__;
	const uid	= 80;
	const gid	= 80;
	
	protected $pool = NULL;
	protected $zmq = NULL;
	public function __construct() {
		$this->pidfile = '/var/run/'.self::pidfile.'.pid';
	}
	private function daemon(){
		if (file_exists($this->pidfile)) {
			echo "The file $this->pidfile exists.\n";
			exit();
		}
		
		$pid = pcntl_fork();
		if ($pid == -1) {
			 die('could not fork');
		} else if ($pid) {
			 // we are the parent
			 //pcntl_wait($status); //Protect against Zombie children
			exit($pid);
		} else {
			// we are the child
			file_put_contents($this->pidfile, getmypid());
			posix_setuid(self::uid);
			posix_setgid(self::gid);
			return(getmypid());
		}
	}
	private function start(){
		$pid = $this->daemon();
		$this->pool = new Pool(self::MAXCONN, \ExampleWorker::class, []);
		$this->zmq = new ZMQSocket(new ZMQContext(), ZMQ::SOCKET_REP);
		$this->zmq->bind(self::LISTEN);
		
		/* Loop receiving and echoing back */
		while ($message = $this->zmq->recv()) {
			//print_r($message);
			//if($trades){
					$this->pool->submit(new Fee($message));
					$this->zmq->send('TRUE');  
			//}else{
			//		$this->zmq->send('FALSE');  
			//}
		}
		$pool->shutdown();	
	}
	private function stop(){

		if (file_exists($this->pidfile)) {
			$pid = file_get_contents($this->pidfile);
			posix_kill($pid, 9); 
			unlink($this->pidfile);
		}
	}
	private function help($proc){
		printf("%s start | stop | help \n", $proc);
	}
	public function main($argv){
		if(count($argv) < 2){
			printf("please input help parameter\n");
			exit();
		}
		if($argv[1] === 'stop'){
			$this->stop();
		}else if($argv[1] === 'start'){
			$this->start();
		}else{
			$this->help($argv[0]);
		}
	}
}

$cgse = new Example();
$cgse->main($argv);
			
			

例 2. 消息隊列與守護進程
			
<?php
declare(ticks = 1);
require_once( __DIR__.'/autoload.class.php' );
umask(077);	
class EDM {
	protected $queue;
	public function __construct() {
		global $argc, $argv;
		$this->argc = $argc;
		$this->argv = $argv;
		$this->pidfile = $this->argv[0].".pid";
		$this->config = new Config('mq');
		$this->logging = new Logging(__DIR__.'/log/'.$this->argv[0].'.'.date('Y-m-d').'.log'); //.H:i:s
		//print_r( $this->config->getArray('mq') );
		//pcntl_signal(SIGHUP, array(&$this,"restart"));
	}
	protected function msgqueue(){
		$exchangeName = 'email'; //交換機名
		$queueName = 'email'; //隊列名
		$routeKey = 'email'; //路由key
		//創(chuàng)建連接和channel
		$connection = new AMQPConnection($this->config->getArray('mq'));
		if (!$connection->connect()) {
			die("Cannot connect to the broker!\n");
		}
		$this->channel = new AMQPChannel($connection);
		$this->exchange = new AMQPExchange($this->channel);
		$this->exchange->setName($exchangeName);
		$this->exchange->setType(AMQP_EX_TYPE_DIRECT); //direct類型
		$this->exchange->setFlags(AMQP_DURABLE); //持久化
		$this->exchange->declare();
		//echo "Exchange Status:".$this->exchange->declare()."\n";
		//創(chuàng)建隊列
		$this->queue = new AMQPQueue($this->channel);
		$this->queue->setName($queueName);
		$this->queue->setFlags(AMQP_DURABLE); //持久化
		$this->queue->declare();
		//echo "Message Total:".$this->queue->declare()."\n";
		//綁定交換機與隊列,并指定路由鍵
		$bind = $this->queue->bind($exchangeName, $routeKey);
		//echo 'Queue Bind: '.$bind."\n";
		//阻塞模式接收消息
		while(true){
			//$this->queue->consume('processMessage', AMQP_AUTOACK); //自動ACK應(yīng)答
			$this->queue->consume(function($envelope, $queue) {
				$msg = $envelope->getBody();
				$queue->ack($envelope->getDeliveryTag()); //手動發(fā)送ACK應(yīng)答
				$this->logging->info('('.'+'.')'.$msg);
				//$this->logging->debug("Message Total:".$this->queue->declare());
			});
			$this->channel->qos(0,1);
			//echo "Message Total:".$this->queue->declare()."\n";
		}
		$conn->disconnect();
	}
	protected function start(){
		if (file_exists($this->pidfile)) {
			printf("%s already running\n", $this->argv[0]);
			exit(0);
		}
		$this->logging->warning("start");
		$pid = pcntl_fork();
		if ($pid == -1) {
			die('could not fork');
		} else if ($pid) {
			//pcntl_wait($status); //等待子進程中斷,防止子進程成為僵尸進程。
			exit(0);
		} else {
			posix_setsid();
			//printf("pid: %s\n", posix_getpid());
			file_put_contents($this->pidfile, posix_getpid());
			
			//posix_kill(posix_getpid(), SIGHUP);
			
			$this->msgqueue();
		}
	}
	protected function stop(){
		if (file_exists($this->pidfile)) {
			$pid = file_get_contents($this->pidfile);
			posix_kill($pid, SIGTERM);
			//posix_kill($pid, SIGKILL);
			unlink($this->pidfile);
			$this->logging->warning("stop");
		}else{
			printf("%s haven't running\n", $this->argv[0]);
		}
	}
	protected function restart(){
		$this->stop();
		$this->start();	
	}
	protected function status(){
		if (file_exists($this->pidfile)) {
			$pid = file_get_contents($this->pidfile);
			printf("%s already running, pid = %s\n", $this->argv[0], $pid);
		}else{
			printf("%s haven't running\n", $this->argv[0]);
		}
	}
	protected function usage(){
		printf("Usage: %s {start | stop | restart | status}\n", $this->argv[0]);
	}
	public function main(){
		//print_r($this->argv);
		if($this->argc != 2){
			$this->usage();
		}else{
			if($this->argv[1] == 'start'){
				$this->start();
			}else if($this->argv[1] == 'stop'){
				$this->stop();
			}else if($this->argv[1] == 'restart'){
				$this->restart();
			}else if($this->argv[1] == 'status'){
				$this->status();
			}else{
				$this->usage();
			}
		}
	}
}
$edm = New EDM();
$edm->main();
			
			

5.1. 程序啟動

下面是程序啟動后進入后臺的代碼

通過進程ID文件來判斷,當前進程狀態(tài),如果進程ID文件存在表示程序在運行中,通過代碼file_exists($this->pidfile)實現(xiàn),但而后進程被kill需要手工刪除該文件才能運行

			
	private function daemon(){
		if (file_exists($this->pidfile)) {
			echo "The file $this->pidfile exists.\n";
			exit();
		}
		
		$pid = pcntl_fork();
		if ($pid == -1) {
			 die('could not fork');
		} else if ($pid) {
			// we are the parent
			//pcntl_wait($status); //Protect against Zombie children
			exit($pid);
		} else {
			// we are the child
			file_put_contents($this->pidfile, getmypid());
			posix_setuid(self::uid);
			posix_setgid(self::gid);
			return(getmypid());
		}
	}
			
			

程序啟動后,父進程會推出,子進程會在后臺運行,子進程權(quán)限從root切換到指定用戶,同時將pid寫入進程ID文件。

5.2. 程序停止

程序停止,只需讀取pid文件,然后調(diào)用posix_kill($pid, 9); 最后將該文件刪除。

			
	private function stop(){

		if (file_exists($this->pidfile)) {
			$pid = file_get_contents($this->pidfile);
			posix_kill($pid, 9); 
			unlink($this->pidfile);
		}
	}
			
			

5.3. 單例模式

所有線程共用數(shù)據(jù)庫連接,在多線程中這個非常重要,如果每個線程建立以此數(shù)據(jù)庫連接在關(guān)閉,這對數(shù)據(jù)庫的開銷是巨大的。

protected function getInstance(){
	return self::$dbh;
}			
			

5.4. 實現(xiàn)優(yōu)雅重啟

所謂優(yōu)雅重啟是指進程不退出的情況加實現(xiàn)重新載入包含重置變量,刷新配置文件,重置日志等等

stop/start 或者 restart都會退出進程,重新啟動,導(dǎo)致進程ID改變,同時瞬間退出導(dǎo)致業(yè)務(wù)閃斷。所以很多守護進程都會提供一個reload功能,者就是所謂的優(yōu)雅重啟。

reload 實現(xiàn)原理是給進程發(fā)送SIGHUP信號,可以通過kill命令發(fā)送 kill -s SIGHUP 64881,也可以通過庫函數(shù)實現(xiàn) posix_kill(posix_getpid(), SIGUSR1);

			
<?php
pcntl_signal(SIGTERM,  function($signo) {
    echo "\n This signal is called. [$signo] \n";
    Status::$state = -1;
});

pcntl_signal(SIGHUP,  function($signo) {
    echo "\n This signal is called. [$signo] \n";
    Status::$state = 1;
	Status::$ini = parse_ini_file('test.ini');
});

class Status{
    public static $state = 0;
	public static $ini = null;
}

$pid = pcntl_fork();
if ($pid == -1) {
    die('could not fork');
}

if($pid) {
    // parent
} else {
	$loop = true;
	Status::$ini = parse_ini_file('test.ini');
    while($loop) {
		print_r(Status::$ini);
        while(true) {
			// Dispatching... 
			pcntl_signal_dispatch();
			if(Status::$state == -1) {
				// Do something and end loop.
				$loop = false;
				break;
			}
			
			if(Status::$state == 1) {
				printf("This program is reload.\r\n");
				Status::$state = 0;
				break;
			}
            echo '.';
            sleep(1);
        }
        echo "\n";
    }
    
    echo "Finish \n";
    exit();
}
			
			

創(chuàng)建配置文件

[root@netkiller pcntl]# cat test.ini 
[db]
host=192.168.0.1
port=3306
			

測試方法,首先運行該守護進程

# php signal.reload.php 
Array
(
    [host] => 192.168.0.1
    [port] => 3306
)
			

現(xiàn)在修改配置文件,增加user=test配置項

[root@netkiller pcntl]# cat test.ini 
[db]
host=192.168.0.1
port=3306
user=test
			

發(fā)送信號,在另一個終端窗口,通過ps命令找到該進程的PID,然后使用kill命令發(fā)送SIGHUP信號,然后再通過ps查看進程,你會發(fā)現(xiàn)進程PID沒有改變

[root@netkiller pcntl]# ps ax | grep reload
64881 pts/0    S      0:00 php -c /srv/php/etc/php-cli.ini signal.reload.php
65073 pts/1    S+     0:00 grep --color=auto reload

[root@netkiller pcntl]# kill -s SIGHUP 64881

[root@netkiller pcntl]# ps ax | grep reload
64881 pts/0    S      0:00 php -c /srv/php/etc/php-cli.ini signal.reload.php
65093 pts/1    S+     0:00 grep --color=auto reload

			

配置文件被重新載入

This signal is called. [1] 
This program is reload.

Array
(
    [host] => 192.168.0.1
    [port] => 3306
    [user] => test
)			
			

優(yōu)雅重啟完成。

6. Example

		
<?php
/*
 * PHP Daemon sample.
 * Home: http://netkiller.github.io
 * Author: netkiller<netkiller@msn.com>
 * 
*/
class Logger {
	
	public function __construct(/*Logging $logger*/) {
	}

	public function logger($type, $message) {
		$log = sprintf ( "%s\t%s\t%s\n", date ( 'Y-m-d H:i:s' ), $type, $message );
		file_put_contents ( sprintf(__DIR__."/../log/sender.%s.log", date ( 'Y-m-d' )), $log, FILE_APPEND );
	}
	
}

final class Signal{	
    public static $signo = 0;
	protected static $ini = null;
	public static function set($signo){
		self::$signo = $signo;
	}
	public static function get(){
		return(self::$signo);
	}
	public static function reset(){
		self::$signo = 0;
	}
}

class Test extends Logger {
	//public static $signal = null;
	
	public function __construct() {
		//self::$signal == null;
	}
	public function run(){
		while(true){
			pcntl_signal_dispatch();
			printf(".");
			sleep(1);
			if(Signal::get() == SIGHUP){
				Signal::reset();
				break;
			}
		}
		printf("\n");
	}
}

class Daemon extends Logger {
	/* config */
	const LISTEN = "tcp://192.168.2.15:5555";
	const pidfile 	= __CLASS__;
	const uid		= 80;
	const gid		= 80;
	const sleep	= 5;

	protected $pool 	= NULL;
	protected $config	= array();

	public function __construct($uid, $gid, $class) {
		$this->pidfile = '/var/run/'.basename(get_class($class), '.php').'.pid';
		//$this->config = parse_ini_file('sender.ini', true); //include_once(__DIR__."/config.php");
		$this->uid = $uid;
		$this->gid = $gid;
		$this->class = $class;
		$this->classname = get_class($class);
		
		$this->signal();
	}
	public function signal(){

		pcntl_signal(SIGHUP,  function($signo) /*use ()*/{
			//echo "\n This signal is called. [$signo] \n";
			printf("The process has been reload.\n");
			Signal::set($signo);
		});

	}
	private function daemon(){
		if (file_exists($this->pidfile)) {
			echo "The file $this->pidfile exists.\n";
			exit();
		}

		$pid = pcntl_fork();
		if ($pid == -1) {
			 die('could not fork');
		} else if ($pid) {
			 // we are the parent
			 //pcntl_wait($status); //Protect against Zombie children
			exit($pid);
		} else {
			file_put_contents($this->pidfile, getmypid());
			posix_setuid(self::uid);
			posix_setgid(self::gid);
			return(getmypid());
		}
	}
	private function run(){

		while(true){
			
			printf("The process begin.\n");
			$this->class->run();
			printf("The process end.\n");
			
		}
	}
	private function foreground(){
		$this->run();
	}
	private function start(){
		$pid = $this->daemon();
		for(;;){
			$this->run();
			sleep(self::sleep);
		}
	}
	private function stop(){

		if (file_exists($this->pidfile)) {
			$pid = file_get_contents($this->pidfile);
			posix_kill($pid, 9);
			unlink($this->pidfile);
		}
	}
	private function reload(){
		if (file_exists($this->pidfile)) {
			$pid = file_get_contents($this->pidfile);
			//posix_kill(posix_getpid(), SIGHUP);
			posix_kill($pid, SIGHUP);
		}
	}	
	private function status(){
		if (file_exists($this->pidfile)) {
			$pid = file_get_contents($this->pidfile);
			system(sprintf("ps ax | grep %s | grep -v grep", $pid));
		}
	}
	private function help($proc){
		printf("%s start | stop | restart | status | foreground | help \n", $proc);
	}
	public function main($argv){

		if(count($argv) < 2){
			$this->help($argv[0]);
			printf("please input help parameter\n");
			exit();
		}
		if($argv[1] === 'stop'){
			$this->stop();
		}else if($argv[1] === 'start'){
			$this->start();
        }else if($argv[1] === 'restart'){
			$this->stop();
            $this->start();
		}else if($argv[1] === 'status'){
			$this->status();
		}else if($argv[1] === 'foreground'){
			$this->foreground();
		}else if($argv[1] === 'reload'){
			$this->reload();
		}else{
			$this->help($argv[0]);
		}
	}
}

$daemon = new Daemon(80,80, new Test());
$daemon->main($argv);
?>
		
		

7. 進程意外退出解決方案

如果是非常重要的進程,必須要保證程序正常運行,一旦出現(xiàn)任何異常退出,都需要做即時做處理。下面的程序可能檢查進程是否異常退出,如果退出便立即啟動。

		
#!/bin/sh

LOGFILE=/var/log/$(basename $0 .sh).log
PATTERN="my.php"
RECOVERY="/path/to/my.php start"

while true
do
        TIMEPOINT=$(date -d "today" +"%Y-%m-%d_%H:%M:%S")
        PROC=$(pgrep -o -f ${PATTERN})
        #echo ${PROC}
        if [ -z "${PROC}" ]; then
		${RECOVERY} >> $LOGFILE
                echo "[${TIMEPOINT}] ${PATTERN} ${RECOVERY}" >> $LOGFILE
                
        #else
                #echo "[${TIMEPOINT}] ${PATTERN} ${PROC}" >> $LOGFILE
        fi
sleep 5
done &
		
		

8. 延伸閱讀

PHP高級編程之消息隊列

PHP高級編程之多線程

posted on 2016-09-01 13:46 思月行云 閱讀(363) 評論(0)  編輯 收藏 引用 所屬分類: PHP
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            国产精品嫩草影院一区二区| 亚洲精品一区二区三区福利| 久久男人av资源网站| 欧美第一黄网免费网站| 亚洲一区视频在线观看视频| 欧美精品久久久久久久久久| 亚洲福利视频一区二区| 久久夜色精品| 欧美一区二区啪啪| 国产精品一区二区在线观看| 亚洲欧美国产不卡| 在线视频欧美一区| 国产精品久久久久久一区二区三区 | 欧美精品一级| 亚洲第一综合天堂另类专| 久久久久国产精品www| 欧美一级艳片视频免费观看| 国产一区二区三区视频在线观看 | 久久夜色精品一区| 久久婷婷人人澡人人喊人人爽 | 亚洲国产精品福利| 久久久www免费人成黑人精品| 国产一区视频观看| 欧美高清视频在线| 欧美黑人在线观看| 中文有码久久| 亚洲影院在线观看| 国产夜色精品一区二区av| 久久蜜桃香蕉精品一区二区三区| 久久精品欧美日韩| 亚洲狼人综合| 亚洲一区二区三区在线| 国产在线成人| 91久久国产综合久久| 欧美视频在线观看| 久久久久久国产精品mv| 免费成人高清视频| 亚洲夜晚福利在线观看| 欧美一区二区三区在线视频| 樱桃国产成人精品视频| 亚洲精品视频在线| 国产亚洲激情| 亚洲欧洲一区二区天堂久久| 国产精品一区二区你懂的| 免费成人黄色| 国产精品福利在线| 欧美二区在线播放| 国产精品资源| 亚洲高清中文字幕| 国产欧美午夜| 99视频在线精品国自产拍免费观看| 国产精品私人影院| 欧美护士18xxxxhd| 国产欧美日韩三区| 99国产精品私拍| 亚洲在线电影| 日韩一级精品视频在线观看| 亚洲欧美一区在线| 亚洲电影在线看| 亚洲少妇自拍| 亚洲人久久久| 欧美在线资源| 亚洲一级黄色片| 免费视频久久| 久久性天堂网| 国产午夜精品福利| 一本色道久久综合亚洲二区三区| 精品不卡一区| 一区二区三区欧美成人| 欧美喷潮久久久xxxxx| 欧美亚洲综合久久| 欧美精品一区三区在线观看| 久久国产精品毛片| 欧美日韩亚洲一区三区 | 在线亚洲美日韩| 亚洲高清网站| 久久成人精品电影| 欧美中文字幕在线视频| 欧美日在线观看| 亚洲黄色精品| 狠狠色狠狠色综合日日小说| 亚洲视频免费在线| 一本久道久久综合狠狠爱| 欧美成人午夜| 欧美电影在线| 国内精品久久久久影院优| 一本一本久久a久久精品牛牛影视| 亚洲精品国产精品国自产在线 | 最新69国产成人精品视频免费| 狠狠久久综合婷婷不卡| 亚洲视频导航| 亚洲永久精品国产| 欧美日韩成人一区| 亚洲久久一区二区| 一本大道久久a久久精二百| 欧美在线地址| 久久激情综合网| 国产精品嫩草99a| 99视频有精品| 午夜精品久久久久久久99水蜜桃 | 午夜国产精品视频免费体验区| 亚洲三级视频在线观看| 你懂的国产精品| 噜噜噜91成人网| 国内久久视频| 男人的天堂亚洲在线| 亚洲国产精品久久久| 亚洲国产精品一区二区尤物区| 美国十次了思思久久精品导航| 亚洲第一级黄色片| 久热爱精品视频线路一| 亚洲国产老妈| 在线亚洲高清视频| 国产农村妇女毛片精品久久麻豆| 先锋影音国产一区| 猫咪成人在线观看| 日韩视频三区| 国产精品久在线观看| 欧美一级二区| 亚洲国产精品嫩草影院| 亚洲色图综合久久| 国产精品久久久久影院色老大| 亚洲视频大全| 欧美国产精品日韩| 亚洲午夜高清视频| 国产丝袜美腿一区二区三区| 久久中文在线| 亚洲激情啪啪| 亚洲欧美日韩一区在线观看| 国产在线国偷精品产拍免费yy| 裸体女人亚洲精品一区| 99热这里只有成人精品国产| 久久久www成人免费精品| 亚洲精品乱码久久久久久久久| 国产精品大片| 老鸭窝亚洲一区二区三区| 中文在线资源观看视频网站免费不卡| 久久久精品日韩| 亚洲男人av电影| 亚洲国产精品尤物yw在线观看| 国产精品久久久久9999| 美女精品在线| 欧美一级电影久久| 一区二区三区成人精品| 美女亚洲精品| 欧美在线影院在线视频| 日韩视频专区| 一区二区亚洲| 国产伦精品一区二区三区在线观看 | 国产精品每日更新| 欧美不卡视频一区发布| 欧美在线一二三四区| 亚洲丝袜av一区| 亚洲激情第一页| 快播亚洲色图| 久久精品盗摄| 欧美制服第一页| 亚洲欧美美女| 亚洲性视频网站| 亚洲一二三级电影| 99这里只有精品| 日韩一级免费| 99人久久精品视频最新地址| 亚洲国产成人91精品| 国产日韩精品一区二区三区在线| 欧美视频在线免费看| 欧美日韩国产在线播放| 欧美精品videossex性护士| 麻豆国产精品777777在线| 久久久精品欧美丰满| 久久成人免费网| 久久精品成人一区二区三区蜜臀 | 久久夜色精品亚洲噜噜国产mv | 久久精品国产99精品国产亚洲性色| 亚洲国产精品免费| 欧美激情第9页| 欧美激情aaaa| 亚洲福利视频一区| 亚洲第一级黄色片| 欧美激情中文字幕乱码免费| 美女免费视频一区| 欧美成人午夜激情在线| 欧美大片在线观看一区| 亚洲高清一区二区三区| 亚洲精品美女久久久久| 欧美成人精精品一区二区频| 欧美呦呦网站| 久久久久久久综合| 久久综合99re88久久爱| 另类尿喷潮videofree| 欧美暴力喷水在线| 欧美日韩第一区| 国产精品美女久久| 久久精品视频免费播放| 免费在线观看日韩欧美| 欧美日韩国产成人精品| 国产精品一区二区久久| 在线观看欧美精品| 亚洲免费电影在线| 亚洲综合日韩中文字幕v在线|