电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術(shù)文章
文章詳情頁

用PHP構(gòu)建一個(gè)簡(jiǎn)易監(jiān)視引擎

瀏覽:51日期:2024-02-10 08:10:38

摘要:在本文中,讓我們共同探討基于PHP語言構(gòu)建一個(gè)基本的服務(wù)器端監(jiān)視引擎的諸多技巧及注意事項(xiàng),并給出完整的源碼實(shí)現(xiàn)。

一. 更改工作目錄的問題

當(dāng)你編寫一個(gè)監(jiān)視程序時(shí),讓它設(shè)置自己的工作目錄通常更好些。這樣以來,如果你使用一個(gè)相對(duì)路徑讀寫文件,那么,它會(huì)根據(jù)情況自動(dòng)處理用戶期望存放文件的位置。總是限制程序中使用的路徑盡管是一種良好的實(shí)踐;但是,卻失去了應(yīng)有的靈活性。因此,改變你的工作目錄的最安全的方法是,既使用chdir()也使用chroot()。

chroot()可用于PHP的CLI和CGI版本中,但是卻要求程序以根權(quán)限運(yùn)行。chroot()實(shí)際上把當(dāng)前進(jìn)程的路徑從根目錄改變到指定的目錄。這使得當(dāng)前進(jìn)程只能執(zhí)行存在于該目錄下的文件。經(jīng)常情況下,chroot()由服務(wù)器作為一個(gè)'安全設(shè)備'使用以確保惡意代碼不會(huì)修改一個(gè)特定的目錄之外的文件。請(qǐng)牢記,盡管chroot()能夠阻止你訪問你的新目錄之外的任何文件,但是,任何當(dāng)前打開的文件資源仍然能夠被存取。例如,下列代碼能夠打開一個(gè)日志文件,調(diào)用chroot()并切換到一個(gè)數(shù)據(jù)目錄;然后,仍然能夠成功地登錄并進(jìn)而打開文件資源:

<?php$logfile = fopen('/var/log/chroot.log', 'w');chroot('/Users/george');fputs($logfile, 'Hello From Inside The Chrootn');?>

如果一個(gè)應(yīng)用程序不能使用chroot(),那么你可以調(diào)用chdir()來設(shè)置工作目錄。例如,當(dāng)代碼需要加載特定的代碼(這些代碼能夠在系統(tǒng)的任何地方被定位時(shí)),這是很有用的。注意,chdir()沒有提供安全機(jī)制來防止打開未授權(quán)的文件。

二. 放棄特權(quán)

當(dāng)編寫Unix守護(hù)程序時(shí),一種經(jīng)典的安全預(yù)防措施是讓它們放棄所有不需要的特權(quán);否則,擁有不需要的特權(quán)容易招致不必要的麻煩。在代碼(或PHP本身)中含有漏洞的情況下,通過確保一個(gè)守護(hù)程序以最小權(quán)限用戶身份運(yùn)行,往往能夠使損失減到最小。

一種實(shí)現(xiàn)此目的的方法是,以非特權(quán)用戶身份執(zhí)行該守護(hù)程序。然而,如果程序需要在一開始就打開非特權(quán)用戶無權(quán)打開的資源(例如日志文件,數(shù)據(jù)文件,套接字,等等)的話,這通常是不夠的。

如果你以根用戶身份運(yùn)行,那么你能夠借助于posix_setuid()和posiz_setgid()函數(shù)來放棄你的特權(quán)。下面的示例把當(dāng)前運(yùn)行程序的特權(quán)改變?yōu)橛脩鬾obody所擁有的那些權(quán)限:

$pw=posix_getpwnam('nobody');posix_setuid($pw['uid']);posix_setgid($pw['gid']);

就象chroot()一樣,任何在放棄特權(quán)之前被打開的特權(quán)資源都會(huì)保持為打開,但是不能創(chuàng)建新的資源。

三. 保證排它性

你可能經(jīng)常想實(shí)現(xiàn):一個(gè)腳本在任何時(shí)刻僅運(yùn)行一個(gè)實(shí)例。為了保護(hù)腳本,這是特別重要的,因?yàn)樵诤笈_(tái)運(yùn)行容易導(dǎo)致偶然情況下調(diào)用多個(gè)實(shí)例。

保證這種排它性的標(biāo)準(zhǔn)技術(shù)是,通過使用flock()來讓腳本鎖定一個(gè)特定的文件(經(jīng)常是一個(gè)加鎖文件,并且被排它式使用)。如果鎖定失敗,該腳本應(yīng)該輸出一個(gè)錯(cuò)誤并退出。下面是一個(gè)示例:

$fp=fopen('/tmp/.lockfile','a');if(!$fp || !flock($fp, LOCK_EX | LOCK_NB)) { fputs(STDERR, 'Failed to acquire lockn'); exit; }/*成功鎖定以安全地執(zhí)行工作*/

注意,有關(guān)鎖機(jī)制的討論涉及較多內(nèi)容,在此不多加解釋。

四. 構(gòu)建監(jiān)視服務(wù)

在這一節(jié)中,我們將使用PHP來編寫一個(gè)基本的監(jiān)視引擎。因?yàn)槟悴粫?huì)事先知道怎樣改變,所以你應(yīng)該使它的實(shí)現(xiàn)既靈活又具可能性。該記錄程序應(yīng)該能夠支持任意的服務(wù)檢查(例如,HTTP和FTP服務(wù))并且能夠以任意方式(通過電子郵件,輸出到一個(gè)日志文件,等等)記錄事件。你當(dāng)然想讓它以一個(gè)守護(hù)程序方式運(yùn)行;所以,你應(yīng)該請(qǐng)求它輸出其完整的當(dāng)前狀態(tài)。

一個(gè)服務(wù)需要實(shí)現(xiàn)下列抽象類:

abstract class ServiceCheck { const FAILURE = 0; const SUCCESS = 1; protected $timeout = 30; protected $next_attempt; protected $current_status = ServiceCheck::SUCCESS; protected $previous_status = ServiceCheck::SUCCESS; protected $frequency = 30; protected $description; protected $consecutive_failures = 0; protected $status_time; protected $failure_time; protected $loggers = array(); abstract public function __construct($params); public function __call($name, $args) {if(isset($this->$name)) { return $this->$name;} } public function set_next_attempt() {$this->next_attempt = time() + $this->frequency; } public abstract function run(); public function post_run($status) {if($status !== $this->current_status) { $this->previous_status = $this->current_status;}if($status === self::FAILURE) { if( $this->current_status === self::FAILURE ) {$this->consecutive_failures++; } else {$this->failure_time = time(); }}else { $this->consecutive_failures = 0;}$this->status_time = time();$this->current_status = $status;$this->log_service_event(); }  public function log_current_status() {foreach($this->loggers as $logger) { $logger->log_current_status($this);} } private function log_service_event() {foreach($this->loggers as $logger) { $logger->log_service_event($this);} } public function register_logger(ServiceLogger $logger) {$this->loggers[] = $logger; }}

上面的__call()重載方法提供對(duì)一個(gè)ServiceCheck對(duì)象的參數(shù)的只讀存取操作:

· timeout-在引擎終止檢查之前,這一檢查能夠掛起多長(zhǎng)時(shí)間。

· next_attempt-下次嘗試連接到服務(wù)器的時(shí)間。

· current_status-服務(wù)的當(dāng)前狀態(tài):SUCCESS或FAILURE。

· previous_status-當(dāng)前狀態(tài)之前的狀態(tài)。

· frequency-每隔多長(zhǎng)時(shí)間檢查一次服務(wù)。

· description-服務(wù)描述。

· consecutive_failures-自從上次成功以來,服務(wù)檢查連續(xù)失敗的次數(shù)。

· status_time-服務(wù)被檢查的最后時(shí)間。

· failure_time-如果狀態(tài)為FAILED,則它代表發(fā)生失敗的時(shí)間。

這個(gè)類還實(shí)現(xiàn)了觀察者模式,允許ServiceLogger類型的對(duì)象注冊(cè)自身,然后當(dāng)調(diào)用log_current_status()或log_service_event()時(shí)調(diào)用它。

這里實(shí)現(xiàn)的關(guān)鍵函數(shù)是run(),它負(fù)責(zé)定義應(yīng)該怎樣執(zhí)行檢查。如果檢查成功,它應(yīng)該返回SUCCESS;否則返回FAILURE。

當(dāng)定義在run()中的服務(wù)檢查返回后,post_run()方法被調(diào)用。它負(fù)責(zé)設(shè)置對(duì)象的狀態(tài)并實(shí)現(xiàn)記入日志。

ServiceLogger接口:指定一個(gè)日志類僅需要實(shí)現(xiàn)兩個(gè)方法:log_service_event()和log_current_status(),它們分別在當(dāng)一個(gè)run()檢查返回時(shí)和當(dāng)實(shí)現(xiàn)一個(gè)普通狀態(tài)請(qǐng)求時(shí)被調(diào)用。

該接口如下所示:

interface ServiceLogger { public function log_service_event(ServiceCheck$service); public function log_current_status(ServiceCheck$service);}

最后,你需要編寫引擎本身。該想法類似于在前一節(jié)編寫簡(jiǎn)單程序時(shí)使用的思想:服務(wù)器應(yīng)該創(chuàng)建一個(gè)新的進(jìn)程來處理每一次檢查并使用一個(gè)SIGCHLD處理器來檢測(cè)當(dāng)檢查完成時(shí)的返回值。可以同時(shí)檢查的最大數(shù)目應(yīng)該是可配置的,從而可以防止對(duì)系統(tǒng)資源的過渡使用。所有的服務(wù)和日志都將在一個(gè)XML文件中定義。

下面是定義該引擎的ServiceCheckRunner類:

class ServiceCheckRunner { private $num_children; private $services = array(); private $children = array(); public function _ _construct($conf, $num_children) {$loggers = array();$this->num_children = $num_children;$conf = simplexml_load_file($conf);foreach($conf->loggers->logger as $logger) { $class = new Reflection_Class('$logger->class'); if($class->isInstantiable()) {$loggers['$logger->id'] = $class->newInstance(); } else {fputs(STDERR, '{$logger->class} cannot be instantiated.n');exit; }}foreach($conf->services->service as $service) { $class = new Reflection_Class('$service->class'); if($class->isInstantiable()) {$item = $class->newInstance($service->params);foreach($service->loggers->logger as $logger) { $item->register_logger($loggers['$logger']);}$this->services[] = $item; } else {fputs(STDERR, '{$service->class} is not instantiable.n');exit; }} } private function next_attempt_sort($a, $b){if($a->next_attempt() == $b->next_attempt()) { return 0;}return ($a->next_attempt() < $b->next_attempt())? -1 : 1; } private function next(){usort($this->services,array($this,'next_attempt_sort'));return $this->services[0]; } public function loop(){declare(ticks=1);pcntl_signal(SIGCHLD, array($this, 'sig_child'));pcntl_signal(SIGUSR1, array($this, 'sig_usr1'));while(1) { $now = time(); if(count($this->children)< $this->num_children) {$service = $this->next();if($now < $service->next_attempt()) { sleep(1); continue;}$service->set_next_attempt();if($pid = pcntl_fork()) { $this->children[$pid] = $service;} else { pcntl_alarm($service->timeout()); exit($service->run());} } }  } public function log_current_status(){foreach($this->services as $service) { $service->log_current_status();} } private function sig_child($signal){$status = ServiceCheck::FAILURE;pcntl_signal(SIGCHLD, array($this, 'sig_child'));while(($pid = pcntl_wait($status, WNOHANG)) > 0){ $service = $this->children[$pid]; unset($this->children[$pid]); if(pcntl_wifexited($status) && pcntl_wexitstatus($status) ==ServiceCheck::SUCCESS)  {$status = ServiceCheck::SUCCESS; } $service->post_run($status);} } private function sig_usr1($signal){pcntl_signal(SIGUSR1, array($this, 'sig_usr1'));$this->log_current_status(); }}

這是一個(gè)很復(fù)雜的類。其構(gòu)造器讀取并分析一個(gè)XML文件,創(chuàng)建所有的將被監(jiān)視的服務(wù),并創(chuàng)建記錄它們的日志程序。

loop()方法是該類中的主要方法。它設(shè)置請(qǐng)求的信號(hào)處理器并檢查是否能夠創(chuàng)建一個(gè)新的子進(jìn)程?,F(xiàn)在,如果下一個(gè)事件(以next_attempt時(shí)間CHUO排序)運(yùn)行良好,那么一個(gè)新的進(jìn)程將被創(chuàng)建。在這個(gè)新的子進(jìn)程內(nèi),發(fā)出一個(gè)警告以防止測(cè)試持續(xù)時(shí)間超出它的時(shí)限,然后執(zhí)行由run()定義的測(cè)試。

還存在兩個(gè)信號(hào)處理器:SIGCHLD處理器sig_child(),負(fù)責(zé)收集已終止的子進(jìn)程并執(zhí)行它們的服務(wù)的post_run()方法;SIGUSR1處理器sig_usr1(),簡(jiǎn)單地調(diào)用所有已注冊(cè)的日志程序的log_current_status()方法,這可以用于得到整個(gè)系統(tǒng)的當(dāng)前狀態(tài)。

當(dāng)然,這個(gè)監(jiān)視架構(gòu)并不沒有做任何實(shí)際的事情。但是首先,你需要檢查一個(gè)服務(wù)。下列這個(gè)類檢查是否你從一個(gè)HTTP服務(wù)器取回一個(gè)'200 Server OK'響應(yīng):

class HTTP_ServiceCheck extends ServiceCheck{ public $url; public function _ _construct($params){foreach($params as $k => $v) { $k = '$k'; $this->$k = '$v';} } public function run(){if(is_resource(@fopen($this->url, 'r'))) { return ServiceCheck::SUCCESS;}else { return ServiceCheck::FAILURE;} }}

與你以前構(gòu)建的框架相比,這個(gè)服務(wù)極其簡(jiǎn)單,在此恕不多描述。

五. 示例ServiceLogger進(jìn)程

下面是一個(gè)示例ServiceLogger進(jìn)程。當(dāng)一個(gè)服務(wù)停用時(shí),它負(fù)責(zé)把一個(gè)電子郵件發(fā)送給一個(gè)待命人員:

class EmailMe_ServiceLogger implements ServiceLogger { public function log_service_event(ServiceCheck$service) {if($service->current_status ==ServiceCheck::FAILURE) { $message = 'Problem with{$service->description()}rn'; mail('oncall@example.com', 'Service Event',$message); if($service->consecutive_failures() > 5) {mail('oncall_backup@example.com', 'Service Event', $message); }} } public function log_current_status(ServiceCheck$service){return; }}

如果連續(xù)失敗五次,那么該進(jìn)程還把一個(gè)消息發(fā)送到一個(gè)備份地址。注意,它并沒有實(shí)現(xiàn)一個(gè)有意義的log_current_status()方法。

無論何時(shí)象如下這樣改變一個(gè)服務(wù)的狀態(tài),你都應(yīng)該實(shí)現(xiàn)一個(gè)寫向PHP錯(cuò)誤日志的ServiceLogger進(jìn)程:

class ErrorLog_ServiceLogger implements ServiceLogger { public function log_service_event(ServiceCheck$service) {if($service->current_status() !==$service->previous_status()) { if($service->current_status() ===ServiceCheck::FAILURE) {$status = 'DOWN'; } else {$status = 'UP'; } error_log('{$service->description()} changed status to $status');} } public function log_current_status(ServiceCheck$service) {error_log('{$service->description()}: $status'); }}

該log_current_status()方法意味著,如果進(jìn)程發(fā)送一個(gè)SIGUSR1信號(hào),它將把其完整的當(dāng)前狀態(tài)復(fù)制到你的PHP錯(cuò)誤日志中。 該引擎使用如下的一個(gè)配置文件:

<config>?。糽oggers><logger>?。糹d>errorlog</id> <class>ErrorLog_ServiceLogger</class></logger><logger>?。糹d>emailme</id> <class>EmailMe_ServiceLogger</class></logger>?。?loggers> <services><service>?。糲lass>HTTP_ServiceCheck</class>?。紁arams><description>OmniTI HTTP Check</description><url>http://www.omniti.com</url><timeout>30</timeout><frequency>900</frequency> </params>?。糽oggers><logger>errorlog</logger><logger>emailme</logger> </loggers></service>?。約ervice>?。糲lass>HTTP_ServiceCheck</class> <params><description>Home Page HTTP Check</description><url>http://www.schlossnagle.org/~george</url><timeout>30</timeout><frequency>3600</frequency>?。?params>?。糽oggers><logger>errorlog</logger>?。?loggers></service></services></config>

當(dāng)傳遞這個(gè)XML文件時(shí),ServiceCheckRunner的構(gòu)造器對(duì)于每一個(gè)指定的日志實(shí)例化一個(gè)日志記錄程序。然后,它相應(yīng)于每一個(gè)指定的服務(wù)實(shí)例化一個(gè)ServiceCheck對(duì)象。

注意 該構(gòu)造器使用Reflection_Class類來實(shí)現(xiàn)該服務(wù)和日志類的內(nèi)在檢查-在你試圖實(shí)例化它們之前。盡管這是不必要的,但是它很好地演示了PHP 5中新的反射(Reflection)API的使用。除了這些類以外,反射API還提供一些類來實(shí)現(xiàn)對(duì)PHP中幾乎任何內(nèi)部實(shí)體(類,方法或函數(shù))的內(nèi)在檢查。

為了使用你構(gòu)建的引擎,你仍然需要一些包裝代碼。監(jiān)視程序應(yīng)該會(huì)禁止你試圖兩次啟動(dòng)它-你不需要對(duì)每一個(gè)事件建立兩份消息。當(dāng)然,該監(jiān)視程序還應(yīng)該接收包括下列選項(xiàng)在內(nèi)的一些選項(xiàng):

選項(xiàng) 描述 [-f] 引擎的配置文件的一個(gè)位置,默認(rèn)是monitor.xml。 [-n]; 引擎允許的子進(jìn)程池的大小,默認(rèn)是5。 [-d]; 一個(gè)停用該引擎的守護(hù)功能的標(biāo)志。在你編寫一個(gè)把信息輸出到stdout或stderr的調(diào)試ServiceLogger進(jìn)程時(shí),這是很有用的。

下面是最終的監(jiān)視程序腳本,它分析選項(xiàng),保證排它性并且運(yùn)行服務(wù)檢查:

require_once 'Service.inc';require_once 'Console/Getopt.php';$shortoptions = 'n:f:d';$default_opts = array('n' => 5, 'f' =>'monitor.xml');$args = getOptions($default_opts, $shortoptions,null);$fp = fopen('/tmp/.lockfile', 'a');if(!$fp || !flock($fp, LOCK_EX | LOCK_NB)) { fputs($stderr, 'Failed to acquire lockn'); exit;}if(!$args['d']) { if(pcntl_fork()) {exit; } posix_setsid(); if(pcntl_fork()) {exit; }}fwrite($fp, getmypid());fflush($fp);$engine = new ServiceCheckRunner($args['f'],$args['n']);$engine->loop();

注意,這個(gè)示例使用了定制的getOptions()函數(shù)。

在編寫一個(gè)適當(dāng)?shù)呐渲梦募螅憧梢园慈缦路绞絾?dòng)該腳本:

> ./monitor.php -f /etc/monitor.xml

這可以保護(hù)并繼續(xù)監(jiān)視直到機(jī)器被關(guān)掉或該腳本被殺死。

這個(gè)腳本相當(dāng)復(fù)雜,但是仍然存在一些容易改進(jìn)的地方,這些只好留給讀者作為練習(xí)之用:

· 添加一個(gè)重新分析配置文件的SIGHUP處理器以便你能夠在不啟動(dòng)服務(wù)器的情況下改變配置。

· 編寫一個(gè)能夠登錄到一個(gè)數(shù)據(jù)庫的ServiceLogger以用于存儲(chǔ)查詢數(shù)據(jù)。

· 編寫一個(gè)Web前端程序以為整個(gè)監(jiān)視系統(tǒng)提供一種良好的GUI。

標(biāo)簽: PHP
主站蜘蛛池模板: 活性氧化铝球|氧化铝干燥剂|分子筛干燥剂|氢氧化铝粉-淄博同心材料有限公司 | 精密五金加工厂-CNC数控车床加工_冲压件|蜗杆|螺杆加工「新锦泰」 | 耐压仪-高压耐压仪|徐吉电气| 全国冰箱|空调|洗衣机|热水器|燃气灶维修服务平台-百修家电 | PCB接线端子_栅板式端子_线路板连接器_端子排生产厂家-置恒电气 喷码机,激光喷码打码机,鸡蛋打码机,手持打码机,自动喷码机,一物一码防伪溯源-恒欣瑞达有限公司 假肢-假肢价格-假肢厂家-河南假肢-郑州市力康假肢矫形器有限公司 | 深圳品牌设计公司-LOGO设计公司-VI设计公司-未壳创意 | 科昊仪器超纯水机系统-可成气相液氮罐-美菱超低温冰箱-西安昊兴生物科技有限公司 | 快速门厂家批发_PVC快速卷帘门_高速门_高速卷帘门-广州万盛门业 快干水泥|桥梁伸缩缝止水胶|伸缩缝装置生产厂家-广东广航交通科技有限公司 | 扬尘监测_扬尘监测系统_带证扬尘监测设备 - 郑州港迪科技有限公司 | 颚式破碎机,圆锥破碎机,制砂机-新乡市德诚机电制造有限公司 | 岛津二手液相色谱仪,岛津10A液相,安捷伦二手液相,安捷伦1100液相-杭州森尼欧科学仪器有限公司 | 胜为光纤光缆_光纤跳线_单模尾纤_光纤收发器_ODF光纤配线架厂家直销_北京睿创胜为科技有限公司 - 北京睿创胜为科技有限公司 | 天助网 - 中小企业全网推广平台_生态整合营销知名服务商_天助网采购优选 | 胜为光纤光缆_光纤跳线_单模尾纤_光纤收发器_ODF光纤配线架厂家直销_北京睿创胜为科技有限公司 - 北京睿创胜为科技有限公司 | 化工ERP软件_化工新材料ERP系统_化工新材料MES软件_MES系统-广东顺景软件科技有限公司 | 自动售货机_无人售货机_专业的自动售货机运营商_免费投放售货机-广州富宏主官网 | 偏心半球阀-电动偏心半球阀-调流调压阀-旋球阀-上欧阀门有限公司 | 深圳公司注册-工商注册代理-注册公司流程和费用_护航财税 | LINK FASHION 童装·青少年装展| 广州展览设计公司_展台设计搭建_展位设计装修公司-众派展览装饰 广州展览制作工厂—[优简]直营展台制作工厂_展会搭建资质齐全 | TPE塑胶原料-PPA|杜邦pom工程塑料、PPSU|PCTG材料、PC/PBT价格-悦诚塑胶 | DWS物流设备_扫码称重量方一体机_快递包裹分拣机_广东高臻智能装备有限公司 | 仓储笼_仓储货架_南京货架_仓储货架厂家_南京货架价格低-南京一品仓储设备制造公司 | 校园气象站_超声波气象站_农业气象站_雨量监测站_风途科技 | 商标转让-商标注册-商标查询-软著专利服务平台 - 赣江万网 | 干洗店加盟_洗衣店加盟_干洗店设备-伊蔻干洗「武汉总部」 | 布袋除尘器|除尘器设备|除尘布袋|除尘设备_诺和环保设备 | 西安标准厂房_陕西工业厂房_西咸新区独栋厂房_长信科技产业园官方网站 | 光泽度计_测量显微镜_苏州压力仪_苏州扭力板手维修-苏州日升精密仪器有限公司 | 环氧乙烷灭菌器_压力蒸汽灭菌器_低温等离子过氧化氢灭菌器 _低温蒸汽甲醛灭菌器_清洗工作站_医用干燥柜_灭菌耗材-环氧乙烷灭菌器_脉动真空压力蒸汽灭菌器_低温等离子灭菌设备_河南省三强医疗器械有限责任公司 | 合肥仿石砖_合肥pc砖厂家_合肥PC仿石砖_安徽旭坤建材有限公司 | 聚氨酯保温钢管_聚氨酯直埋保温管道_聚氨酯发泡保温管厂家-沧州万荣防腐保温管道有限公司 | 莱州网络公司|莱州网站建设|莱州网站优化|莱州阿里巴巴-莱州唯佳网络科技有限公司 | 工业淬火油烟净化器,北京油烟净化器厂家,热处理油烟净化器-北京众鑫百科 | 水厂污泥地磅|污泥处理地磅厂家|地磅无人值守称重系统升级改造|地磅自动称重系统维修-河南成辉电子科技有限公司 | 专业生产动态配料系统_饲料配料系统_化肥配料系统等配料系统-郑州鑫晟重工机械有限公司 | 热处理温控箱,热处理控制箱厂家-吴江市兴达电热设备厂 | 作文导航网_作文之家_满分作文_优秀作文_作文大全_作文素材_最新作文分享发布平台 | bng防爆挠性连接管-定做金属防爆挠性管-依客思防爆科技 | 齿轮减速机_齿轮减速电机-VEMT蜗轮蜗杆减速机马达生产厂家瓦玛特传动瑞环机电 | 乐之康护 - 专业护工服务平台,提供医院陪护-居家照护-居家康复 |