03 基于 java udp 做一个dns服务器 和 一个dns代理服务器

前言

这个也是 来自于一个朋友的需求

最终的目的是实现一个 dns 代理服务器, 当然 这本质也是一个 dns 服务器

并且 dns 代理服务器是依赖于 一个 dns 服务器的, 因此 顺便给一个 dns 服务器的 demo

这里 主要是 基于 udp 的一个 dns 请求, 响应数据的交互

dns 服务器

一个基础的 dns 服务器 demo 如下

为了方便测试, 可以将 端口改为 53, 然后使用 System.out.println 之类的进行调试

复制代码
package com.hx.test15;

import org.xbill.DNS.*;

import java.io.ByteArrayOutputStream;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.nio.ByteBuffer;

/**
 * @author Jerry.X.He <970655147@qq.com>
 * @version 1.0
 * @date 2024-07-02 22:33
 */
public class Test25SimpleDnsServer {

    public static void main(String[] args) throws Exception {
        DatagramSocket socket = new DatagramSocket(8053);
        System.out.println("DNS Server listening on port 8053...");

        while (true) {
            byte[] receiveData = new byte[512];
            DatagramPacket requestPacket = new DatagramPacket(receiveData, receiveData.length);
            socket.receive(requestPacket);

            byte[] data = requestPacket.getData();
            int offset = requestPacket.getOffset();
            int length = requestPacket.getLength();

            Message request = new Message(ByteBuffer.wrap(data, offset, length));
            Message response = handleRequest(request);

            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            DNSOutput out = new DNSOutput();
            baos.write(out.toByteArray());
            baos.write(response.toWire());
            byte[] responseData = baos.toByteArray();

            DatagramPacket responsePacket = new DatagramPacket(responseData, responseData.length,
                    requestPacket.getAddress(), requestPacket.getPort());
            socket.send(responsePacket);
        }
    }

    private static Message handleRequest(Message request) throws Exception {
        Message response = new Message(request.getHeader().getID());
        response.getHeader().setFlag(Flags.QR);
        response.getHeader().setOpcode(Opcode.QUERY);

        Record question = request.getQuestion();
        if (question.getName().toString().startsWith("www.abc.com") && question.getType() == Type.A) {
            ARecord answer = new ARecord(question.getName(), DClass.IN, 300, InetAddress.getByName("172.0.0.2"));
            response.addRecord(answer, Section.ANSWER);
        } else {
            // apply default
        }

        return response;
    }


}

最终实现效果如下, dns 服务器中仅仅配置了 "www.abc.com" -> "172.0.0.2"

因此 www.abc.com 能够拿到查询结果, www.baidu.com 拿不到查询结果

dns 代理服务器

主需求是实现一个 dns 的代理服务器

实现如下, 无非就是 代理发送一次 dns 的请求 到目标 dns 服务器, 拿到响应之后 响应给客户端

复制代码
package com.hx.test15;

import org.xbill.DNS.Message;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.nio.ByteBuffer;

/**
 * Test25JavaDnsForwardServer
 *
 * @author Jerry.X.He <970655147@qq.com>
 * @version 1.0
 * @date 2024-07-02 22:10
 */
public class Test25JavaDnsForwardServer {

    public static void main(String[] args) throws IOException {
        DatagramSocket serverSocket = new DatagramSocket(53);
        System.out.println("DNS Proxy Server listening on port 53...");

        while (true) {
            byte[] receiveData = new byte[512];
            DatagramPacket requestPacket = new DatagramPacket(receiveData, receiveData.length);
            serverSocket.receive(requestPacket);

            Message request = readMessage(requestPacket);
            Message response = proxyRequest(request);

            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            baos.write(response.toWire());
            byte[] responseData = baos.toByteArray();

            DatagramPacket responsePacket = new DatagramPacket(responseData, responseData.length,
                    requestPacket.getAddress(), requestPacket.getPort());
            serverSocket.send(responsePacket);
        }
    }

    private static Message readMessage(DatagramPacket packet) throws IOException {
        byte[] data = packet.getData();
        int offset = packet.getOffset();
        int length = packet.getLength();
        return new Message(ByteBuffer.wrap(data, offset, length));
    }

    private static Message proxyRequest(Message request) throws IOException {
        DatagramSocket upstreamSocket = new DatagramSocket();
        InetAddress upstreamAddress = InetAddress.getByName("127.0.0.1");
        int upstreamPort = 8053;

        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        baos.write(request.toWire());
        byte[] requestData = baos.toByteArray();
        DatagramPacket upstreamRequest = new DatagramPacket(requestData, requestData.length, upstreamAddress, upstreamPort);
        upstreamSocket.send(upstreamRequest);

        byte[] upstreamResponseData = new byte[512];
        DatagramPacket upstreamResponse = new DatagramPacket(upstreamResponseData, upstreamResponseData.length);
        upstreamSocket.receive(upstreamResponse);

        Message response = new Message(ByteBuffer.wrap(upstreamResponseData, upstreamResponse.getOffset(), upstreamResponse.getLength()));
        upstreamSocket.close();
        return response;
    }


}

最终实现效果如下, dns 服务器中仅仅配置了 "www.abc.com" -> "172.0.0.2"

因此 www.abc.com 能够拿到查询结果, www.baidu.com 拿不到查询结果

如果 关闭 目标dns服务, 代理服务器会阻塞住

相关推荐
DemonAvenger几秒前
Go网络编程基础:网络模型与协议栈概述
网络协议·架构·go
2501_9160074738 分钟前
iOS 接口频繁请求导致流量激增?抓包分析定位与修复全流程
websocket·网络协议·tcp/ip·http·网络安全·https·udp
Net_Walke1 小时前
【网络协议】WebSocket简介
网络·websocket·网络协议
2501_916013741 小时前
用Fiddler中文版抓包工具掌控微服务架构中的接口调试:联合Postman与Charles的高效实践
websocket·网络协议·tcp/ip·http·网络安全·https·udp
00后程序员张3 小时前
调试 WebView 接口时间戳签名问题:一次精细化排查和修复过程
websocket·网络协议·tcp/ip·http·网络安全·https·udp
小锋学长生活大爆炸4 小时前
【知识】RPC和gRPC
服务器·网络协议·rpc
小高不会迪斯科4 小时前
MIT 6.824学习心得(2) 浅谈多线程和RPC
网络·网络协议·rpc
前端小巷子4 小时前
跨域问题解决方案:CORS(跨域资源共享)
前端·网络协议·面试
帽儿山的枪手13 小时前
为什么Linux需要3种NAT地址转换?一探究竟
linux·网络协议·安全
兴达易控10 天前
Modbus TCP转Profibus DP网关接JF-600MT称重变送器到西门子S7-300plc系统
网络协议