你的位置:首页 > Java教程

[Java教程]基于nodejs的DNS查询工具


  开始这个实例之前,我们简单谈一下Node.js吧,Node.js是一个由JavaScript书写而成的强大Web开发框架,它让开发强壮的、伸缩性良好的服务器端Web应用变得更加简单、容易。这种技术诞生于09年末,在一个JavaScript大会上宣布,当时这项在服务器端运行JavaScript技术让所有参会者惊奇,当时这位提出者给出了一个“hello world”的程序。

1 var http = require('http');2 var server = http.createServer(function(req,res){3   res.writeHead(200);4   res.end('Hello World');5 });6 server.listen(3000,'127.0.0.1');7 console.log("please run at 127.0.0.1:3000");

  于是在地址栏输入127.0.0.1:3000后就出现了我们非常熟悉的“Hello World”

  

  好,简单介绍下Node.js,接下来就开始我们的正题,简单搭建一个DNS查询工具

  准备前期:确保已经安装Node.js,网络上面有很多安装教程,在这里就不详细多说了

(1)需要加载的Node.js原生模块:

  1. Node.js的HTTP模块,用于创建Web的HTTP服务器
  2. Node.js的dns模块,用于DNS域名解析
  3. Node.js的fileSystem模块,用于读取HTML页面
  4. Node.js的querystring模块,处理请求参数
1 var http = require('http'),2   dns = require('dns'),3   fs = require('fs');4   url = require('url');//处理文件url路径5   querystring = require('querystring');//处理前端传回的字符串解析

(2)添加HTTP服务器代码,返回显示index.html(PS:如果是返回html数据,则Content-Type类型值为text/html)

1 http.createServer(function(req,res){2   var pathname = url.parse(req.url).pathname;3   req.setEncoding("utf8");4   res.writeHead(200,{'Content-Type':'text/html'});5   router(res,req,pathname);6 }).listen(3000,'127.0.0.1'); //监听地址为127.0.0.1:30007 console.log("You can try it at 127.0.0.1:3000");

解析出请求中的地址后,用一个路由处理器判断区分用户是要显示index.html主页,还是要进行/parse解析操作,这里的路由控制器就是router方法
(3)处理url路由

1 function router(res,req,pathname){2   switch(pathname){3     case "/parse":parseDns(res,req);break;  //解析url地址4     default:goIndex(res,req);         //显示index主页5   }6 }  

  • goIndex(res,req)函数,主要用于显示一个index.html页面,参数:res  为响应客户端请求对象;req  为客户端请求对象
1 function goIndex(res,req){2   var readPath = __dirname + '/' + url.parse('index.html').pathname;3   var indexPage = fs.readFileSync(readPath);//读取html数据,存放在indexpage变量之中4   res.end(indexPage);              //响应html数据到客户端5 }

  • parseDns(res,req)函数,主要用于解析客户端传递来的域名,并且返回该域名对应的IP地址
 1 1 function parseDns(res,req){ 2 2   var postData = ""; 3 3   req.addListener("data",function(postDataChunk){ 4 4     postData += postDataChunk; 5 5   }); 6 6   req.addListener("end",function(){ 7 7     var resData = getDns(postData,function(domain,addresses){ 8 8       res.writeHead(200,{'Content-Type':'text/html'}); 9 9       res.end("</html><head><meta http-equiv='content-type' content='text/html;charset=utf-8'></head><body><div style='text-align:center'>Domain:<span style='color:red'>" + domain + "</span><br/>IP:<span style='color:red'>" + addresses.join(',') + "</span></div></body></html>");10 10       return;11 11     })12 12   })13 13 }14 14 15 15 function getDns(postData,callback){16 16   var domain = querystring.parse(postData).search_dns;17 17   console.log(postData);           //search_dns=www.qq.com18 18   console.log(querystring.parse(postData));//{ search_dns: 'www.qq.com' }19 19   console.log(domain);           //www.qq.com/20 20   dns.resolve(domain,function(err,addresses){ //返回的addresses是一个数组21 21     if(!addresses){22 22       addresses=['不存在域名'];23 23     }24 24     callback(domain,addresses);25 25   })26 26 }

注意:由于dns.resolve()方法是一个异步执行函数,如果想使用它执行的结果,需要有一个回调函数,并把结果作为参数传入回调函数,才可以传递到函数外面!

(4)主页index.html代码

 1 <!DOCTYPE html> 2 <html> 3 <head> 4   <title>DNS查询</title> 5   <meta http-equiv="content-type" content="text/html;charset=utf-8"> 6 </head> 7 <body> 8   <h1 style='text-align:center'>DNS查询工具</h1> 9   <div style='text-align:center'>10     <form action="/parse" method="post">11       查询DNS:<input type='text' name='search_dns'/>12       <input type='submit' value='查询'/>13     </form>14   </div>15 </body>

  到了这里,已经实现了DNS查询功能了,我把(1)(2)(3)(4)步的代码都放在一个叫parse_dns.js的文件夹里面,在同一根目录文件下存放(5)步的index.html,然后再命令行窗口下找到当前目录(一开始我是用很笨的方法用cd命令找到文件根目录,后来发现sublimeText的Side Bar插件可以直接在所选文件夹右键点击“Open/Run”,直接打开一个叫Window PowerShell的蓝色窗口,功能和cmd窗口一样,在那里已经帮你自动找好文件路径,非常方便),然后直接输入node parse_dns.js,即可显示"You can try it at 127.0.0.1:3000"

在浏览器地址栏输入“127.0.0.1:3000”,就有下图界面

输入“www.baidu.com”进行查询,页面就跳到了“127.0.0.1:3000/parse”

 

  进行到这一步已经是基本完成了,但是笔者并没有将这个系统进行模块化,比如分成服务器创建模块、路由处理模块、逻辑控制模块、错误处理模块等,要想让程序更加健壮并且项目可维护性高,这个是必不可少的,由于是刚学node不久,日后必定写上更好的版本。

 

思考:

(1)这个小工具还存在一些小问题,就是如果我输入类似“https://www.baidu.com/”这样子的格式,程序会解析不成功,而返回“域名不存在”,所以日后我需要引入Node.js的原生模块url,用于url的分解

 

(2)笔者输入“www.qq.com”解析出来的IP,放在浏览器地址栏里打开但并不是腾讯的主页,然而输入“www.baidu.com”解析出来的IP却可以呈现度娘的搜索首页,笔者的室友说是因为有多个服务器解析出来的多个IP,所以这个IP可能不是我正在看到的腾讯主页,还有一个解释是因为端口问题,百度搜索首页是默认80端口,而腾讯首页不是,所以要知道端口号并加到IP后面才可以访问,问题还没搞清楚,所以笔者还会继续研究~