|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575 |
- using MQTTnet.Exceptions;
- using MQTTnet.Packets;
- using MQTTnet.Protocol;
- using System;
- using System.IO;
- using System.Linq;
- using System.Text;
-
- namespace MQTTnet.Serializer
- {
- public sealed class MqttPacketSerializer : IMqttPacketSerializer
- {
- private static byte[] ProtocolVersionV311Name { get; } = Encoding.UTF8.GetBytes("MQTT");
- private static byte[] ProtocolVersionV310Name { get; } = Encoding.UTF8.GetBytes("MQIsdp");
-
- public MqttProtocolVersion ProtocolVersion { get; set; } = MqttProtocolVersion.V311;
-
- public ArraySegment<byte> Serialize(MqttBasePacket packet)
- {
- if (packet == null) throw new ArgumentNullException(nameof(packet));
-
- using (var stream = new MemoryStream(128))
- {
- // Leave enough head space for max header size (fixed + 4 variable remaining length)
- stream.Position = 5;
- var fixedHeader = SerializePacket(packet, stream);
-
- stream.Position = 1;
- var remainingLength = MqttPacketWriter.EncodeRemainingLength((int)stream.Length - 5, stream);
-
- var headerSize = remainingLength + 1;
- var headerOffset = 5 - headerSize;
-
- // Position cursor on correct offset on beginining of array (has leading 0x0)
- stream.Position = headerOffset;
-
- stream.WriteByte(fixedHeader);
-
- #if NET461 || NET452 || NETSTANDARD2_0
- var buffer = stream.GetBuffer();
- #else
- var buffer = stream.ToArray();
- #endif
- return new ArraySegment<byte>(buffer, headerOffset, (int)stream.Length - headerOffset);
- }
- }
-
- private byte SerializePacket(MqttBasePacket packet, Stream stream)
- {
- switch (packet)
- {
- case MqttConnectPacket connectPacket: return Serialize(connectPacket, stream);
- case MqttConnAckPacket connAckPacket: return Serialize(connAckPacket, stream);
- case MqttDisconnectPacket _: return SerializeEmptyPacket(MqttControlPacketType.Disconnect);
- case MqttPingReqPacket _: return SerializeEmptyPacket(MqttControlPacketType.PingReq);
- case MqttPingRespPacket _: return SerializeEmptyPacket(MqttControlPacketType.PingResp);
- case MqttPublishPacket publishPacket: return Serialize(publishPacket, stream);
- case MqttPubAckPacket pubAckPacket: return Serialize(pubAckPacket, stream);
- case MqttPubRecPacket pubRecPacket: return Serialize(pubRecPacket, stream);
- case MqttPubRelPacket pubRelPacket: return Serialize(pubRelPacket, stream);
- case MqttPubCompPacket pubCompPacket: return Serialize(pubCompPacket, stream);
- case MqttSubscribePacket subscribePacket: return Serialize(subscribePacket, stream);
- case MqttSubAckPacket subAckPacket: return Serialize(subAckPacket, stream);
- case MqttUnsubscribePacket unsubscribePacket: return Serialize(unsubscribePacket, stream);
- case MqttUnsubAckPacket unsubAckPacket: return Serialize(unsubAckPacket, stream);
- default: throw new MqttProtocolViolationException("Packet type invalid.");
- }
- }
-
- public MqttBasePacket Deserialize(MqttPacketHeader header, Stream stream)
- {
- if (header == null) throw new ArgumentNullException(nameof(header));
- if (stream == null) throw new ArgumentNullException(nameof(stream));
-
- switch (header.ControlPacketType)
- {
- case MqttControlPacketType.Connect: return DeserializeConnect(stream);
- case MqttControlPacketType.ConnAck: return DeserializeConnAck(stream);
- case MqttControlPacketType.Disconnect: return new MqttDisconnectPacket();
- case MqttControlPacketType.Publish: return DeserializePublish(stream, header);
- case MqttControlPacketType.PubAck: return DeserializePubAck(stream);
- case MqttControlPacketType.PubRec: return DeserializePubRec(stream);
- case MqttControlPacketType.PubRel: return DeserializePubRel(stream);
- case MqttControlPacketType.PubComp: return DeserializePubComp(stream);
- case MqttControlPacketType.PingReq: return new MqttPingReqPacket();
- case MqttControlPacketType.PingResp: return new MqttPingRespPacket();
- case MqttControlPacketType.Subscribe: return DeserializeSubscribe(stream, header);
- case MqttControlPacketType.SubAck: return DeserializeSubAck(stream, header);
- case MqttControlPacketType.Unsubscibe: return DeserializeUnsubscribe(stream, header);
- case MqttControlPacketType.UnsubAck: return DeserializeUnsubAck(stream);
- default: throw new MqttProtocolViolationException($"Packet type ({(int)header.ControlPacketType}) not supported.");
- }
- }
-
- private static MqttBasePacket DeserializeUnsubAck(Stream stream)
- {
- return new MqttUnsubAckPacket
- {
- PacketIdentifier = stream.ReadUInt16()
- };
- }
-
- private static MqttBasePacket DeserializePubComp(Stream stream)
- {
- return new MqttPubCompPacket
- {
- PacketIdentifier = stream.ReadUInt16()
- };
- }
-
- private static MqttBasePacket DeserializePubRel(Stream stream)
- {
- return new MqttPubRelPacket
- {
- PacketIdentifier = stream.ReadUInt16()
- };
- }
-
- private static MqttBasePacket DeserializePubRec(Stream stream)
- {
- return new MqttPubRecPacket
- {
- PacketIdentifier = stream.ReadUInt16()
- };
- }
-
- private static MqttBasePacket DeserializePubAck(Stream stream)
- {
- return new MqttPubAckPacket
- {
- PacketIdentifier = stream.ReadUInt16()
- };
- }
-
- private static MqttBasePacket DeserializeUnsubscribe(Stream stream, MqttPacketHeader header)
- {
- var packet = new MqttUnsubscribePacket
- {
- PacketIdentifier = stream.ReadUInt16(),
- };
-
- while (stream.Position != header.BodyLength)
- {
- packet.TopicFilters.Add(stream.ReadStringWithLengthPrefix());
- }
-
- return packet;
- }
-
- private static MqttBasePacket DeserializeSubscribe(Stream stream, MqttPacketHeader header)
- {
- var packet = new MqttSubscribePacket
- {
- PacketIdentifier = stream.ReadUInt16()
- };
-
- while (stream.Position != header.BodyLength)
- {
- packet.TopicFilters.Add(new TopicFilter(
- stream.ReadStringWithLengthPrefix(),
- (MqttQualityOfServiceLevel)stream.ReadByte()));
- }
-
- return packet;
- }
-
- private static MqttBasePacket DeserializePublish(Stream stream, MqttPacketHeader mqttPacketHeader)
- {
- var fixedHeader = new ByteReader(mqttPacketHeader.FixedHeader);
- var retain = fixedHeader.Read();
- var qualityOfServiceLevel = (MqttQualityOfServiceLevel)fixedHeader.Read(2);
- var dup = fixedHeader.Read();
-
- var topic = stream.ReadStringWithLengthPrefix();
-
- ushort? packetIdentifier = null;
- if (qualityOfServiceLevel > MqttQualityOfServiceLevel.AtMostOnce)
- {
- packetIdentifier = stream.ReadUInt16();
- }
-
- var packet = new MqttPublishPacket
- {
- PacketIdentifier = packetIdentifier,
- Retain = retain,
- Topic = topic,
- Payload = stream.ReadRemainingData(mqttPacketHeader),
- QualityOfServiceLevel = qualityOfServiceLevel,
- Dup = dup
- };
-
- return packet;
- }
-
- private static MqttBasePacket DeserializeConnect(Stream stream)
- {
- stream.ReadBytes(2); // Skip 2 bytes for header and remaining length.
-
- MqttProtocolVersion protocolVersion;
- var protocolName = stream.ReadBytes(4);
-
- if (protocolName.SequenceEqual(ProtocolVersionV311Name))
- {
- protocolVersion = MqttProtocolVersion.V311;
- }
- else
- {
- protocolName = protocolName.Concat(stream.ReadBytes(2)).ToArray();
- if (protocolName.SequenceEqual(ProtocolVersionV310Name))
- {
- protocolVersion = MqttProtocolVersion.V310;
- }
- else
- {
- throw new MqttProtocolViolationException("Protocol name is not supported.");
- }
- }
-
- stream.ReadByte(); // Skip protocol level
- var connectFlags = stream.ReadByte();
-
- var connectFlagsReader = new ByteReader(connectFlags);
- connectFlagsReader.Read(); // Reserved.
-
- var packet = new MqttConnectPacket
- {
- ProtocolVersion = protocolVersion,
- CleanSession = connectFlagsReader.Read()
- };
-
- var willFlag = connectFlagsReader.Read();
- var willQoS = connectFlagsReader.Read(2);
- var willRetain = connectFlagsReader.Read();
- var passwordFlag = connectFlagsReader.Read();
- var usernameFlag = connectFlagsReader.Read();
-
- packet.KeepAlivePeriod = stream.ReadUInt16();
- packet.ClientId = stream.ReadStringWithLengthPrefix();
-
- if (willFlag)
- {
- packet.WillMessage = new MqttApplicationMessage
- {
- Topic = stream.ReadStringWithLengthPrefix(),
- Payload = stream.ReadWithLengthPrefix(),
- QualityOfServiceLevel = (MqttQualityOfServiceLevel)willQoS,
- Retain = willRetain
- };
- }
-
- if (usernameFlag)
- {
- packet.Username = stream.ReadStringWithLengthPrefix();
- }
-
- if (passwordFlag)
- {
- packet.Password = stream.ReadStringWithLengthPrefix();
- }
-
- ValidateConnectPacket(packet);
- return packet;
- }
-
- private static MqttBasePacket DeserializeSubAck(Stream stream, MqttPacketHeader header)
- {
- var packet = new MqttSubAckPacket
- {
- PacketIdentifier = stream.ReadUInt16()
- };
-
- while (stream.Position != header.BodyLength)
- {
- packet.SubscribeReturnCodes.Add((MqttSubscribeReturnCode)stream.ReadByte());
- }
-
- return packet;
- }
-
- private MqttBasePacket DeserializeConnAck(Stream stream)
- {
- var packet = new MqttConnAckPacket();
-
- var firstByteReader = new ByteReader(stream.ReadByte());
-
- if (ProtocolVersion == MqttProtocolVersion.V311)
- {
- packet.IsSessionPresent = firstByteReader.Read();
- }
-
- packet.ConnectReturnCode = (MqttConnectReturnCode)stream.ReadByte();
-
- return packet;
- }
-
- private static void ValidateConnectPacket(MqttConnectPacket packet)
- {
- if (packet == null) throw new ArgumentNullException(nameof(packet));
-
- if (string.IsNullOrEmpty(packet.ClientId) && !packet.CleanSession)
- {
- throw new MqttProtocolViolationException("CleanSession must be set if ClientId is empty [MQTT-3.1.3-7].");
- }
- }
-
- private static void ValidatePublishPacket(MqttPublishPacket packet)
- {
- if (packet == null) throw new ArgumentNullException(nameof(packet));
-
- if (packet.QualityOfServiceLevel == 0 && packet.Dup)
- {
- throw new MqttProtocolViolationException("Dup flag must be false for QoS 0 packets [MQTT-3.3.1-2].");
- }
- }
-
- private byte Serialize(MqttConnectPacket packet, Stream stream)
- {
- ValidateConnectPacket(packet);
-
- // Write variable header
- if (ProtocolVersion == MqttProtocolVersion.V311)
- {
- stream.WriteWithLengthPrefix(ProtocolVersionV311Name);
- stream.WriteByte(0x04); // 3.1.2.2 Protocol Level 4
- }
- else
- {
- stream.WriteWithLengthPrefix(ProtocolVersionV310Name);
- stream.WriteByte(0x03); // Protocol Level 3
- }
-
- var connectFlags = new ByteWriter(); // 3.1.2.3 Connect Flags
- connectFlags.Write(false); // Reserved
- connectFlags.Write(packet.CleanSession);
- connectFlags.Write(packet.WillMessage != null);
-
- if (packet.WillMessage != null)
- {
- connectFlags.Write((int)packet.WillMessage.QualityOfServiceLevel, 2);
- connectFlags.Write(packet.WillMessage.Retain);
- }
- else
- {
- connectFlags.Write(0, 2);
- connectFlags.Write(false);
- }
-
- if (packet.Password != null && packet.Username == null)
- {
- throw new MqttProtocolViolationException("If the User Name Flag is set to 0, the Password Flag MUST be set to 0 [MQTT-3.1.2-22].");
- }
-
- connectFlags.Write(packet.Password != null);
- connectFlags.Write(packet.Username != null);
-
- stream.Write(connectFlags);
- stream.Write(packet.KeepAlivePeriod);
- stream.WriteWithLengthPrefix(packet.ClientId);
-
- if (packet.WillMessage != null)
- {
- stream.WriteWithLengthPrefix(packet.WillMessage.Topic);
- stream.WriteWithLengthPrefix(packet.WillMessage.Payload);
- }
-
- if (packet.Username != null)
- {
- stream.WriteWithLengthPrefix(packet.Username);
- }
-
- if (packet.Password != null)
- {
- stream.WriteWithLengthPrefix(packet.Password);
- }
-
- return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.Connect);
- }
-
- private byte Serialize(MqttConnAckPacket packet, Stream stream)
- {
- if (ProtocolVersion == MqttProtocolVersion.V310)
- {
- stream.WriteByte(0);
- }
- else if (ProtocolVersion == MqttProtocolVersion.V311)
- {
- var connectAcknowledgeFlags = new ByteWriter();
- connectAcknowledgeFlags.Write(packet.IsSessionPresent);
- stream.Write(connectAcknowledgeFlags);
- }
- else
- {
- throw new MqttProtocolViolationException("Protocol version not supported.");
- }
-
- stream.WriteByte((byte)packet.ConnectReturnCode);
-
- return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.ConnAck);
- }
-
- private static byte Serialize(MqttPubRelPacket packet, Stream stream)
- {
- if (!packet.PacketIdentifier.HasValue)
- {
- throw new MqttProtocolViolationException("PubRel packet has no packet identifier.");
- }
-
- stream.Write(packet.PacketIdentifier.Value);
-
- return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.PubRel, 0x02);
- }
-
- private static byte Serialize(MqttPublishPacket packet, Stream stream)
- {
- ValidatePublishPacket(packet);
-
- stream.WriteWithLengthPrefix(packet.Topic);
-
- if (packet.QualityOfServiceLevel > MqttQualityOfServiceLevel.AtMostOnce)
- {
- if (!packet.PacketIdentifier.HasValue)
- {
- throw new MqttProtocolViolationException("Publish packet has no packet identifier.");
- }
-
- stream.Write(packet.PacketIdentifier.Value);
- }
- else
- {
- if (packet.PacketIdentifier > 0)
- {
- throw new MqttProtocolViolationException("Packet identifier must be empty if QoS == 0 [MQTT-2.3.1-5].");
- }
- }
-
- if (packet.Payload?.Length > 0)
- {
- stream.Write(packet.Payload, 0, packet.Payload.Length);
- }
-
- byte fixedHeader = 0;
-
- if (packet.Retain)
- {
- fixedHeader |= 0x01;
- }
-
- fixedHeader |= (byte)((byte)packet.QualityOfServiceLevel << 1);
-
- if (packet.Dup)
- {
- fixedHeader |= 0x08;
- }
-
- return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.Publish, fixedHeader);
- }
-
- private static byte Serialize(MqttPubAckPacket packet, Stream stream)
- {
- if (!packet.PacketIdentifier.HasValue)
- {
- throw new MqttProtocolViolationException("PubAck packet has no packet identifier.");
- }
-
- stream.Write(packet.PacketIdentifier.Value);
-
- return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.PubAck);
- }
-
- private static byte Serialize(MqttPubRecPacket packet, Stream stream)
- {
- if (!packet.PacketIdentifier.HasValue)
- {
- throw new MqttProtocolViolationException("PubRec packet has no packet identifier.");
- }
-
- stream.Write(packet.PacketIdentifier.Value);
-
- return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.PubRec);
- }
-
- private static byte Serialize(MqttPubCompPacket packet, Stream stream)
- {
- if (!packet.PacketIdentifier.HasValue)
- {
- throw new MqttProtocolViolationException("PubComp packet has no packet identifier.");
- }
-
- stream.Write(packet.PacketIdentifier.Value);
-
- return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.PubComp);
- }
-
- private static byte Serialize(MqttSubscribePacket packet, Stream stream)
- {
- if (!packet.TopicFilters.Any()) throw new MqttProtocolViolationException("At least one topic filter must be set [MQTT-3.8.3-3].");
-
- if (!packet.PacketIdentifier.HasValue)
- {
- throw new MqttProtocolViolationException("Subscribe packet has no packet identifier.");
- }
-
- stream.Write(packet.PacketIdentifier.Value);
-
- if (packet.TopicFilters?.Count > 0)
- {
- foreach (var topicFilter in packet.TopicFilters)
- {
- stream.WriteWithLengthPrefix(topicFilter.Topic);
- stream.WriteByte((byte)topicFilter.QualityOfServiceLevel);
- }
- }
-
- return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.Subscribe, 0x02);
- }
-
- private static byte Serialize(MqttSubAckPacket packet, Stream stream)
- {
- if (!packet.PacketIdentifier.HasValue)
- {
- throw new MqttProtocolViolationException("SubAck packet has no packet identifier.");
- }
-
- stream.Write(packet.PacketIdentifier.Value);
-
- if (packet.SubscribeReturnCodes?.Any() == true)
- {
- foreach (var packetSubscribeReturnCode in packet.SubscribeReturnCodes)
- {
- stream.WriteByte((byte)packetSubscribeReturnCode);
- }
- }
-
- return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.SubAck);
- }
-
- private static byte Serialize(MqttUnsubscribePacket packet, Stream stream)
- {
- if (!packet.TopicFilters.Any()) throw new MqttProtocolViolationException("At least one topic filter must be set [MQTT-3.10.3-2].");
-
- if (!packet.PacketIdentifier.HasValue)
- {
- throw new MqttProtocolViolationException("Unsubscribe packet has no packet identifier.");
- }
-
- stream.Write(packet.PacketIdentifier.Value);
-
- if (packet.TopicFilters?.Any() == true)
- {
- foreach (var topicFilter in packet.TopicFilters)
- {
- stream.WriteWithLengthPrefix(topicFilter);
- }
- }
-
- return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.Unsubscibe, 0x02);
- }
-
- private static byte Serialize(MqttUnsubAckPacket packet, Stream stream)
- {
- if (!packet.PacketIdentifier.HasValue)
- {
- throw new MqttProtocolViolationException("UnsubAck packet has no packet identifier.");
- }
-
- stream.Write(packet.PacketIdentifier.Value);
- return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.UnsubAck);
- }
-
- private static byte SerializeEmptyPacket(MqttControlPacketType type)
- {
- return MqttPacketWriter.BuildFixedHeader(type);
- }
- }
- }
|