我有一个 CoAP 服务器和客户端,我用 Java 编写了 Californium 库。我可以通过客户端或带有 Copper(Cu) 扩展名的 Firefox 从服务器获取资源。我想获得从服务器获得的响应的总大小。我已经可以获取有效负载的大小,但我想要总消息大小。我是通过代码还是使用工具来完成这一点并不重要。到目前为止,我一直无法在 Google 上找到实现此目的的方法。
1 回答
这取决于你想得到什么。
1)如果你只是想发现一些足迹。
我建议使用wireshark,捕获和分析数据包:
2)如果您想以编程方式使用长度
据我所知,没有干净直接的方法可以实现这一目标。
有一些解决方法。
连接器
您可以包装一个连接器并显式设置客户端的端点(以及服务器,但我显示的是客户端版本):
CoapClient client = new CoapClient(new URI(uri));
client.setEndpoint(
new CoapEndpoint(
new InterceptingUDPConnector(
new UDPConnector(
new InetSocketAddress(
portNumber
)
)
),
NetworkConfig.getStandard()
)
)
这是连接器包装器:
public class InterceptingUDPConnector implements Connector {
private final Connector origin;
public InterceptingUDPConnector(Connector origin) {
this.origin = origin;
}
@Override
public void send(RawData msg) {
System.out.println("Length:" + msg.getSize());
origin.send(msg);
}
@Override
public void setRawDataReceiver(RawDataChannel messageHandler) {
RawDataChannel decodingChannel = raw -> {
System.out.println("Length: " + raw.getSize())
messageHandler.receiveData(raw);
};
origin.setRawDataReceiver(decodingChannel);
}
@Override
public void start() throws IOException {
origin.start();
}
@Override
public void stop() { origin.stop();
}
@Override
public void destroy() { origin.destroy();
}
@Override
public InetSocketAddress getAddress() {
return origin.getAddress();
}
}
然而,很难将这些长度和响应联系起来。
我使用以下类来解析 RawData:
import org.eclipse.californium.core.coap.Message;
import org.eclipse.californium.core.network.serialization.DataParser;
import org.eclipse.californium.elements.RawData;
public class ParsedPacket {
private final RawData packet;
public ParsedPacket(RawData packet) {
this.packet = packet;
}
@Override
public String toString() {
Message msg = null;
DataParser parser = new DataParser(packet.getBytes());
if (parser.isEmpty()) {
msg = parser.parseEmptyMessage();
} else if (parser.isRequest()) {
msg = parser.parseRequest();
} else if (parser.isResponse()) {
msg = parser.parseResponse();
}
return (msg == null) ? "" : msg.toString();
}
}
这样,您可以使用例如 MID 和令牌来关联特定响应和总长度。
这里的一个陷阱是您必须为此使用一些全局存储。
我不建议将其用于生产。除了一些罕见的情况,例如打印一些数据包信息+长度。这也可能通过启用一些适当的 Californium 记录器级别来涵盖。
因此,从字面上看,这是一种不好的做法。
MessageTracer 和序列化消息
您可以使用 MessageLengthTracer 类实现org.eclipse.californium.core.network.interceptors.MessageInterceptor接口。
client.setEndpoint(...);
client.endpoint().setMessageTracer(new MessageLengthTracer());
在那里,您可以使用org.eclipse.californium.core.network.serialization.DataSerializer使用与以前的解决方法类似的技术来序列化消息并获取它们的长度。
不过,这也是一种解决方法,并且存在两个陷阱 - 可能的长度不准确和双重序列化工作。
PS如果有人说我错了,我会很高兴并指出我这样做的干净方法。