向 "router.bittorrent.com" 发送 DHT 查询响应乱码

Send DHT queries to "router.bittorrent.com" response garbled text

我阅读了 bep_0005 页中的 DHT 协议。

但是当我发送 ping 查询或 find_node 查询时,服务器响应乱码文本(router.bittorrent.com:6881dht.transmissionbt.com:6881)

下面是Java源代码

    public String ping(final String id) {
    System.out.println("Start ping:" + id);
    Bencode bencode = new Bencode();
    byte[] encoded = bencode.encode(new HashMap<Object, Object>() {
        private static final long serialVersionUID = 4225164001818744013L;

        {
            put("t", "tr");
            put("y", "q");
            put("q", "ping");
            put("a", new HashMap<Object, Object>() {
                private static final long serialVersionUID = -6092073963971093460L;

                {
                    put("id", id);
                }
            });
        }
    });
    byte[] result = client.send(new String(encoded, bencode.getCharset()));
    Map<String, Object> dict = bencode.decode(result, Type.DICTIONARY);
    System.out.println("Bdecoded Data:" + dict);
    return "";
}

发送数据包

ping 查询 = {"t":"aa", "y":"q", "q":"ping", "a" :{"id":"abcdefghij0123456789"}}

编码 = d1:ad2:id20:abcdefghij0123456789e1:q4:ping1:t2:aa1:y1:qe

添加到 bep_0005 协议的响应如下:

响应 = {"t":"aa", "y":"r", "r": {"id":"mnopqrstuvwxyz123456" }}

编码 = d1:rd2:id20:mnopqrstuvwxyz123456e1:t2:aa1:y1:re

但是我的回复是:

响应 = {ip=��P����, r={id=2��NisQ��J��)ͺ����F|��g}, t=tr, y=r}

bencoded = d2:ip6:��P����1:rd2:id20:2��NisQ��J��)ͺ����F|��ge1:t2:tr1:y1:re

发送udp部分Java代码为:

    public byte[] send(String sendData) {
    DatagramSocket client;
    try {
        client = new DatagramSocket();
        client.setSoTimeout(5000);
        byte[] sendBuffer;
        sendBuffer = sendData.getBytes();
        InetAddress addr = InetAddress.getByName("router.bittorrent.com");
        int port = 6881;
        DatagramPacket sendPacket = new DatagramPacket(sendBuffer, sendBuffer.length, addr, port);
        client.send(sendPacket);
        byte[] receiveBuf = new byte[512];
        DatagramPacket receivePacket = new DatagramPacket(receiveBuf, receiveBuf.length);
        client.receive(receivePacket);
        System.out.println("Client Source Data:" + Arrays.toString(receivePacket.getData()));
        String receiveData = new String(receivePacket.getData(), "UTF-8");
        System.out.println("Client String Data:" + receiveData);
        client.close();
        return receivePacket.getData();
    } catch (SocketException e) {
        e.printStackTrace();
    } catch (UnknownHostException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }
    return null;
}

读取UTF-8格式的响应,但是iso-8859-1也是乱码。

谁能帮帮我,谢谢!

the server response a garbled text

否,响应经过编码并包含原始二进制数据。
它不能被视为文本。

在 BEP5 中,要使示例中的原始二进制文件 node_id 可打印, 它被巧妙地选择为仅包含字母数字字符。
参见:

ip 键是一个扩展,解释如下:BEP42 - DHT Security extension

收到的回复完全有效。

TODO: Working Java code

Map<String, Object> dict = bencode.decode(result, Type.DICTIONARY);

这会为您提供消息的解码根字典 Map。在其中,您应该找到 r 字典作为另一个映射,并在该映射中找到 id 值。 id 的类型取决于您使用的 bedecoding 库。

如果它是 ByteBufferbyte[] 那么你应该有 20 个字节,如果你需要它是人类可读的,你可以十六进制编码(到 40 个字符)。 DHT 协议处理原始哈希值,而不是十六进制值。

如果它是 String,那么在对它进行十六进制编码之前,您必须将字符串转换回 byte[]。这只有在 bdecoder 使用 ISO 8859-1 解码时才有可能,因为该字符集是往返安全的,而 utf-8 不适用于任意字节序列。