深入浅出ghostbuster剖析NodeJS与PhantomJS的通讯机制
深入浅出ghostbuster剖析NodeJS与PhantomJS的通讯机制
蔡建良 2013-11-14
一. 让我们开始吧
通过命令行来执行
1) 进行命令窗口: cmd
2) 进入resources-requested.js 所在目录:
cd 你的目录\ghostbuster\ghostbuster-master\tests
3) 执行nodejs代码: node resources-requested.js
执行成功后,会在tests目录下会生成一个google.png图片。
resources-requested.js
这是一个nodejs的主文件,调用ghostbuster.js模块,用于对网页进行操作,代码如下:

ghostbuster.js
这是一个nodejs的文件,导出一个spawn方法。代码如下。

bridge.js
这是一个phantomjs的文件,代码如下。

代码下载: http://download.csdn.net/detail/janehlp/6552571
二. 代码解析
ghostbuster.spawn方法传入两个参数,一个是端口号8089,另一个是匿名的回调函数function(phantom){….},回调函数在ghostbuster.spawn方法内部被调用,调用时机是在socket连接监听事件被触发时发生。如下图所示:

谁来触发socket连接事件,这个当然是socket客户端。这是通讯最核心的地方。如果连接事件无法被触发,那回调函数就不会被执行。nodejs与phantomjs也无法进行通讯。
三. 触发socket连接事件的真相
1) ghostbuster.spawn方法通过child_process模块的spawn方法调用phantomjs命令来执行bridge.js代码。phantomjs命令是一个c++写的exe文件,该文件目录必须在环境变量PATH中。

2) spawn(“phantomjs”,[bridge,port]执行全过程
这里的port是传入bridge.js中的参数。

注意:bridge.js代码是由phantomjs解析执行,而ghostbuster.js是由nodejs解析执行,不要搞混了。两者的内置模块是不相同的,不能混合使用。
webpage、fs、system是phantomjs内置模块。
bridge.js代码首先先会创建一个页面变量controlpage,并打开http://127.0.0.1:8089网址。
这个就好像你用浏览器打开一个新的页面,并在地址栏中输入网址http://127.0.0.1:8089。
由于前面ghostbuster.js已创建了一个http服务,监听端口是8089。因此http://127.0.0.1:8089由ghostbuster.js中的http服务来响应,响应代码如下图:

ghostbuster.js中的http服务响应http://127.0.0.1:8089请求,并返回一个text/html格式的网页内容。网页内容来自conrolPage变量,该变量由getControlPage()方法赋值。
getControlPage()方法返回什么呢?
nodejs与phantomjs通过websocket来进行通讯。它们之间沟通的桥梁客户端页面,客户端页面相当于一个client.html文件。这个文件的内容如下:

getControlPage()方法就是返回上面这个client.html的内容,方法如下图所示:

还是回到bridge.js代码中来,当phantomjs执行bridge.js,会生成一个类似client.html的页面。
而这个页面中采用了socket.io来与服务端建立websocket连接,从而实现了与nodejs的通讯。

client.html执行如下脚本与node服务端连接,连接成功后会触发connection事件,回调函数也会被执行。

注意:ghostbuster.js就是所谓的服务端。
我们再来看看ghostbuster.js中连接事件被触发后回调函数的样子。

四. 页面创建完整过程
callback(p)执行的是resources-requested.js中的回调方法function(phantom){…}。
function(phantom){…}方法使用ghostbuster中p对象的createWebPage来创建页面。
操作网页之前必须先创建一个空白页面,然后再打开网址并进行其它操作。

function(phantom)这个方法中的参数phantom是ghostbuster.js中的变量p。p这个对象封装了phantomjs命令。让我们看看对象p封装了哪些方法。

对象p封装了3个属性和5个方法。其中createWebPage是用于创建页面。
让我们更深入了解一下创建页面的完整过程:
1) resources-requested.js执行p对象中的createWebPage方法。

2) ghostbusert.js执行p. createWebPage方法,并调用了request方法。
参数说明: properties为{settings:{loadImages:true}},callback为function(page){…}。

3) ghostbusert.js执行request方法,调用socket发出cmd命令。
参数说明:args为[“createWebPage”,properties],callback为function(page){…},cbId为空。
args.splice(0,0,cbId); 是指从第0个位置开始插入 cbId,即回调函数的索引。
将回调函数缓存到requests数组,后面responseHandler方法还需要用到。

4) client.html页面接收到cmd命令,并执行alert(msg)方法。
参数说明: msg为[cbId,“createWebPage”,properties]

5) bridge.js监听controlpage页面的alert事件,并执行createWebPage(msg)方法。
参数说明: message为[cbId,“createWebPage”,properties]

6) bridge.js执行createWebPage(msg)方法创建页面对象,并调用respond方法返回信息。
参数说明: req为[cbId,“createWebPage”,properties]数组对象。
req[0]为cbId。index为页面索引。

7) bridge.js执行respond(args)方法通过页面的evaluate方法发送socket命令向nodejs服务端发送响应信息。
参数说明: args为[cbId,“createWebPage”,index]

8) ghostbuster.js通过socket.on(“res”,responseHandler)对res命令设置监听处理responseHandler。
参数说明:response为字符串,值为[cbId,“createWebPage”,index]。
cb为resources-requested.js中的回调函数function(page) {…}。res为页面索引index。

WebPage方法在ghostbuster.js中定义如下:

9) resources-requested.js的回调函数function(page) {…}。
终于又回到主程序来了,这时我们可以通过WebPage对象来对页面做具体的业务功能,如打开一个页面进行截屏等操作。

五. 页面创建流程图

深入浅出ghostbuster剖析NodeJS与PhantomJS的通讯机制的更多相关文章
- v78.01 鸿蒙内核源码分析(消息映射篇) | 剖析LiteIpc(下)进程通讯机制 | 百篇博客分析OpenHarmony源码
百篇博客分析|本篇为:(消息映射篇) | 剖析LiteIpc(下)进程通讯机制 进程通讯相关篇为: v26.08 鸿蒙内核源码分析(自旋锁) | 当立贞节牌坊的好同志 v27.05 鸿蒙内核源码分析( ...
- 深入剖析C/C++函数的参数传递机制
2014-07-29 20:16 深入剖析C/C++函数的参数传递机制 C语言的函数入口参数,可以使用值传递和指针传递方式,C++又多了引用(reference)传递方式.引用传递方式在使用上类 ...
- 【工业串口和网络软件通讯平台(SuperIO)教程】八.SuperIO通讯机制与设备驱动对接的说明
SuperIO相关资料下载:http://pan.baidu.com/s/1pJ7lZWf 1.1 通讯机制说明 通讯的总体机制采用呼叫应答方式,就是上位机软件主动发送请求数据命令,下位机终端接 ...
- 【工业串口和网络软件通讯平台(SuperIO)教程】一.通讯机制
1.1 应用场景 通讯平台的交互对象包括两方面:第一.与硬件产品交互.第二.与软件产品交互.基本这两方面考虑,通讯平台一般会应用在两个场景: 1)通讯平台应用在PC机上 主要应用在自动站的工控机 ...
- <MFC_1>深入剖析MFC的WinMain和消息机制
一.开篇引论 熟悉Win32开发的朋友,应该非常了解它的基本组成和流程 1. WinMain:书写窗口类(WNDCLASS) -> 注册窗口类 -> 创建窗口 -> 显示窗口和更新窗 ...
- ActiveMQ之 TCP通讯机制
ActiveMQ支持多种通讯协议TCP/UDP等,我们选取最常用的TCP来分析ActiveMQ的通讯机制.首先我们来明确一个概念: 客户(Client):消息的生产者.消费者对ActiveMQ来说都 ...
- 一篇文章了解相见恨晚的 Android Binder 进程间通讯机制【转】
本文转载自:https://blog.csdn.net/freekiteyu/article/details/70082302 Android-Binder进程间通讯机制 概述 最近在学习Binder ...
- 深度剖析java中JDK动态代理机制
https://www.jb51.net/article/110342.htm 本篇文章主要介绍了深度剖析java中JDK动态代理机制 ,动态代理避免了开发人员编写各个繁锁的静态代理类,只需简单地指定 ...
- 面试题: nodejs 的事件轮询机制
setTimeout(function(){ console.log('setTimeout()执行了') },0) setImmediate(function(){ console.log('set ...
随机推荐
- 如何修改Linux系统的TTL值
在网络中,黑客如果用ping命令去探测 一个主机,根据TTL基数可以推测操作系统的类型.对于一个没有经过任何网关和路由的网络, 直接ping对方系统得到的TTL值,被叫做"TTL基数&qu ...
- React-非dom属性-dangerouslySetInnerHTML标签
<!DOCTYPE html> <html lang="zh-cn"> <head> <meta charset="UTF-8& ...
- JTAG的SWD接线方式
JTAG引脚示意图: 一.SWD 和传统的调试方式区别 1. SWD 模式比 JTAG 在高速模式下面更加可靠 2. GPIO 刚好缺一个的时候, 可以使用 SWD 仿真, 这种模式支持更少的引脚 3 ...
- Android 使用全局变量传递数据
使用全局变量传递数据,所谓的全局变量类似于jee开发中的application变量.申明后,全局调用.只有当内存被清理后,才被销毁.否则一直可以调用. 还是使用点击一个button,传递一个数据到另一 ...
- c# 串行【序列化】和解串【反序列化】
C# 串行[序列化]和解串[反序列化] 一. 什么是序列化和反序列话呢? 相信我们做程序的都会遇到这种情况,需要将C#中某一个结构很复杂的类的对象存储起来,或者通过网路传输到远程的客户端程序中去, ...
- 在linux系统下检查postgresql数据库安装,登录数据库及简单的查看数据库
1. 检查Linux系统是否安装数据库 首先查看自己的系统是否安装了postgresql数据库命令如下: rpm -qa | grep postgresql 如果没有显示查询结果(如下图所示)说 ...
- Entity Framework Architecture
http://www.entityframeworktutorial.net/EntityFramework-Architecture.aspx The following figure shows ...
- [Golang]实习最后一天小纪念+并发爬虫小练习
今天是我在公司实习的最后一天,一个月的时间真的是太短暂了,我非常享受在公司工作的这一个月,在这里Leader和同事们对我的帮助极大地促进了我技术水平的进步和自信心的提升,我发自内心地感谢白山云科技给我 ...
- 总结Selenium自动化测试方法(六)常见的异常错误处理
六.常见的异常错误处理 NoSuchElementException: Message: Unable to locate element: {"method":"xpa ...
- Restrict each user to a single session in window server 2008 R2 or 2012
Restrict each user to a single session in window server 2008 R2 or 2012 2014-10-31 In window server ...