zoukankan      html  css  js  c++  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:3000
    7 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.com
    18 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后面才可以访问,问题还没搞清楚,所以笔者还会继续研究~

  • 相关阅读:
    Yii2框架之旅(三)
    Yii2框架之旅(二)
    Redis本地集群搭建(5版本以上)
    Redis入门笔记
    Java如何使用elasticsearch进行模糊查询--转载
    springboot集成elasticsearch7.6.1,常用api调用,创建,查找,删除索引,crud,高亮。。。--转载
    SpringBoot整合Elasticsearch7.2.0的实现方法-转载
    Spring Webflux 入门 -转载
    java 视频流截屏,形成缩略图
    记录一下 spring boot 线程处理返回数据
  • 原文地址:https://www.cnblogs.com/DTBelieve/p/5346641.html
Copyright © 2011-2022 走看看