package org.opendaylight.controller.netconf.util.messages;
-import com.google.common.base.Charsets;
-import com.google.common.base.Optional;
-import com.google.common.collect.Lists;
-import io.netty.buffer.Unpooled;
-import io.netty.channel.ChannelHandler;
-import io.netty.handler.codec.DelimiterBasedFrameDecoder;
+import java.io.ByteArrayInputStream;
+import java.io.IOException;
+import java.nio.ByteBuffer;
+import java.util.Arrays;
+import java.util.List;
+
import org.opendaylight.controller.netconf.api.NetconfDeserializerException;
import org.opendaylight.controller.netconf.api.NetconfMessage;
import org.opendaylight.controller.netconf.util.xml.XmlUtil;
import org.w3c.dom.Document;
import org.xml.sax.SAXException;
-import java.io.ByteArrayInputStream;
-import java.io.IOException;
-import java.nio.ByteBuffer;
-import java.util.Arrays;
-import java.util.List;
+import com.google.common.base.Charsets;
+import com.google.common.base.Optional;
+import com.google.common.collect.Lists;
/**
* NetconfMessageFactory for (de)serializing DOM documents.
private static final Logger logger = LoggerFactory.getLogger(NetconfMessageFactory.class);
- public static final byte[] endOfMessage = "]]>]]>".getBytes(Charsets.UTF_8);
-
- public static final byte[] endOfChunk = "\n##\n".getBytes(Charsets.UTF_8);
-
- private static final int MAX_CHUNK_SIZE = 1024; // Bytes
-
- private FramingMechanism framing = FramingMechanism.EOM;
-
private final Optional<String> clientId;
public NetconfMessageFactory() {
this.clientId = clientId;
}
- public static ChannelHandler getDelimiterFrameDecoder() {
- return new DelimiterBasedFrameDecoder(Integer.MAX_VALUE, Unpooled.copiedBuffer(endOfMessage));
- }
-
@Override
- public List<NetconfMessage> parse(byte[] bytes) throws DeserializerException, DocumentedException {
- String s = Charsets.UTF_8.decode(ByteBuffer.wrap(bytes)).toString();
- logger.debug("Parsing message \n{}", s);
- if (bytes[0] == '[') {
- // yuma sends auth information in the first line. Ignore until ]\n
- // is found.
- int endOfAuthHeader = ByteArray.findByteSequence(bytes, new byte[] { ']', '\n' });
+ public NetconfMessage parse(byte[] bytes) throws DeserializerException, DocumentedException {
+ logMessage(bytes);
+
+ String additionalHeader = null;
+
+ if (startsWithAdditionalHeader(bytes)) {
+ // Auth information containing username, ip address... extracted for monitoring
+ int endOfAuthHeader = getAdditionalHeaderEndIndex(bytes);
if (endOfAuthHeader > -1) {
+ byte[] additionalHeaderBytes = Arrays.copyOfRange(bytes, 0, endOfAuthHeader + 2);
+ additionalHeader = additionalHeaderToString(additionalHeaderBytes);
bytes = Arrays.copyOfRange(bytes, endOfAuthHeader + 2, bytes.length);
}
}
- List<NetconfMessage> messages = Lists.newArrayList();
+ NetconfMessage message;
try {
Document doc = XmlUtil.readXmlToDocument(new ByteArrayInputStream(bytes));
- messages.add(new NetconfMessage(doc));
+ message = new NetconfMessage(doc, additionalHeader);
} catch (final SAXException | IOException | IllegalStateException e) {
throw new NetconfDeserializerException("Could not parse message from " + new String(bytes), e);
}
- return messages;
+ return message;
+ }
+
+ private int getAdditionalHeaderEndIndex(byte[] bytes) {
+ for (String possibleEnd : Lists.newArrayList("]\n", "]\r\n")) {
+ int idx = ByteArray.findByteSequence(bytes, possibleEnd.getBytes(Charsets.UTF_8));
+
+ if (idx != -1) {
+ return idx;
+ }
+ }
+
+ return -1;
+ }
+
+ private boolean startsWithAdditionalHeader(byte[] bytes) {
+ List<String> possibleStarts = Lists.newArrayList("[", "\r\n[", "\n[");
+ for (String possibleStart : possibleStarts) {
+ int i = 0;
+ for (byte b : possibleStart.getBytes(Charsets.UTF_8)) {
+ if(bytes[i]!=b)
+ break;
+
+ return true;
+ }
+ }
+
+ return false;
+ };
+
+ private void logMessage(byte[] bytes) {
+ String s = Charsets.UTF_8.decode(ByteBuffer.wrap(bytes)).toString();
+ logger.debug("Parsing message \n{}", s);
+ }
+
+ private String additionalHeaderToString(byte[] bytes) {
+ return Charsets.UTF_8.decode(ByteBuffer.wrap(bytes)).toString();
}
@Override
Comment comment = netconfMessage.getDocument().createComment("clientId:" + clientId.get());
netconfMessage.getDocument().appendChild(comment);
}
- byte[] bytes = (this.framing == FramingMechanism.EOM) ? this.putEOM(netconfMessage) : this
- .putChunked(netconfMessage);
+ ByteBuffer msgBytes;
+ if(netconfMessage.getAdditionalHeader().isPresent()) {
+ String header = netconfMessage.getAdditionalHeader().get();
+ logger.trace("Header of netconf message parsed \n{}", header);
+ msgBytes = Charsets.UTF_8.encode(header + xmlToString(netconfMessage.getDocument()));
+ } else {
+ msgBytes = Charsets.UTF_8.encode(xmlToString(netconfMessage.getDocument()));
+ }
String content = xmlToString(netconfMessage.getDocument());
logger.trace("Putting message \n{}", content);
- return bytes;
- }
-
- private byte[] putEOM(NetconfMessage msg) {
- // create byte buffer from the String XML
- // all Netconf messages are encoded using UTF-8
- final ByteBuffer msgBytes = Charsets.UTF_8.encode(xmlToString(msg.getDocument()));
- final ByteBuffer result = ByteBuffer.allocate(msgBytes.limit() + endOfMessage.length);
- result.put(msgBytes);
- // put end of message
- result.put(endOfMessage);
- return result.array();
- }
-
- private byte[] putChunked(NetconfMessage msg) {
- final ByteBuffer msgBytes = Charsets.UTF_8.encode(xmlToString(msg.getDocument()));
- final NetconfMessageHeader h = new NetconfMessageHeader();
- if (msgBytes.limit() > MAX_CHUNK_SIZE)
- logger.warn("Netconf message too long, should be split.");
- h.setLength(msgBytes.limit());
- final byte[] headerBytes = h.toBytes();
- final ByteBuffer result = ByteBuffer.allocate(headerBytes.length + msgBytes.limit() + endOfChunk.length);
- result.put(headerBytes);
- result.put(msgBytes);
- result.put(endOfChunk);
- return result.array();
+ byte[] b = new byte[msgBytes.limit()];
+ msgBytes.get(b);
+ return b;
}
private String xmlToString(Document doc) {
return XmlUtil.toString(doc, false);
}
-
- /**
- * For Hello message the framing is always EOM, but the framing mechanism
- * may change.
- *
- * @param fm
- * new framing mechanism
- */
- public void setFramingMechanism(final FramingMechanism fm) {
- logger.debug("Framing mechanism changed to {}", fm);
- this.framing = fm;
- }
}