MqttV310PacketFormatter.cs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604
  1. using System;
  2. using System.Linq;
  3. using MQTTnet.Adapter;
  4. using MQTTnet.Exceptions;
  5. using MQTTnet.Packets;
  6. using MQTTnet.Protocol;
  7. namespace MQTTnet.Formatter.V3
  8. {
  9. public class MqttV310PacketFormatter : IMqttPacketFormatter
  10. {
  11. private const int FixedHeaderSize = 1;
  12. private static readonly MqttPingReqPacket PingReqPacket = new MqttPingReqPacket();
  13. private static readonly MqttPingRespPacket PingRespPacket = new MqttPingRespPacket();
  14. private static readonly MqttDisconnectPacket DisconnectPacket = new MqttDisconnectPacket();
  15. private readonly IMqttPacketWriter _packetWriter;
  16. public MqttV310PacketFormatter()
  17. : this(new MqttPacketWriter())
  18. {
  19. }
  20. public MqttV310PacketFormatter(IMqttPacketWriter packetWriter)
  21. {
  22. _packetWriter = packetWriter;
  23. }
  24. public IMqttDataConverter DataConverter { get; } = new MqttV310DataConverter();
  25. public ArraySegment<byte> Encode(MqttBasePacket packet)
  26. {
  27. if (packet == null) throw new ArgumentNullException(nameof(packet));
  28. // Leave enough head space for max header size (fixed + 4 variable remaining length = 5 bytes)
  29. _packetWriter.Reset(5);
  30. _packetWriter.Seek(5);
  31. var fixedHeader = EncodePacket(packet, _packetWriter);
  32. var remainingLength = (uint)(_packetWriter.Length - 5);
  33. var remainingLengthSize = MqttPacketWriter.GetLengthOfVariableInteger(remainingLength);
  34. var headerSize = FixedHeaderSize + remainingLengthSize;
  35. var headerOffset = 5 - headerSize;
  36. // Position cursor on correct offset on beginning of array (has leading 0x0)
  37. _packetWriter.Seek(headerOffset);
  38. _packetWriter.Write(fixedHeader);
  39. _packetWriter.WriteVariableLengthInteger(remainingLength);
  40. var buffer = _packetWriter.GetBuffer();
  41. return new ArraySegment<byte>(buffer, headerOffset, _packetWriter.Length - headerOffset);
  42. }
  43. public MqttBasePacket Decode(ReceivedMqttPacket receivedMqttPacket)
  44. {
  45. if (receivedMqttPacket == null) throw new ArgumentNullException(nameof(receivedMqttPacket));
  46. var controlPacketType = receivedMqttPacket.FixedHeader >> 4;
  47. if (controlPacketType < 1 || controlPacketType > 14)
  48. {
  49. throw new MqttProtocolViolationException($"The packet type is invalid ({controlPacketType}).");
  50. }
  51. switch ((MqttControlPacketType)controlPacketType)
  52. {
  53. case MqttControlPacketType.Connect: return DecodeConnectPacket(receivedMqttPacket.Body);
  54. case MqttControlPacketType.ConnAck: return DecodeConnAckPacket(receivedMqttPacket.Body);
  55. case MqttControlPacketType.Disconnect: return DisconnectPacket;
  56. case MqttControlPacketType.Publish: return DecodePublishPacket(receivedMqttPacket);
  57. case MqttControlPacketType.PubAck: return DecodePubAckPacket(receivedMqttPacket.Body);
  58. case MqttControlPacketType.PubRec: return DecodePubRecPacket(receivedMqttPacket.Body);
  59. case MqttControlPacketType.PubRel: return DecodePubRelPacket(receivedMqttPacket.Body);
  60. case MqttControlPacketType.PubComp: return DecodePubCompPacket(receivedMqttPacket.Body);
  61. case MqttControlPacketType.PingReq: return PingReqPacket;
  62. case MqttControlPacketType.PingResp: return PingRespPacket;
  63. case MqttControlPacketType.Subscribe: return DecodeSubscribePacket(receivedMqttPacket.Body);
  64. case MqttControlPacketType.SubAck: return DecodeSubAckPacket(receivedMqttPacket.Body);
  65. case MqttControlPacketType.Unsubscibe: return DecodeUnsubscribePacket(receivedMqttPacket.Body);
  66. case MqttControlPacketType.UnsubAck: return DecodeUnsubAckPacket(receivedMqttPacket.Body);
  67. default: throw new MqttProtocolViolationException($"Packet type ({controlPacketType}) not supported.");
  68. }
  69. }
  70. public void FreeBuffer()
  71. {
  72. _packetWriter.FreeBuffer();
  73. }
  74. private byte EncodePacket(MqttBasePacket packet, IMqttPacketWriter packetWriter)
  75. {
  76. switch (packet)
  77. {
  78. case MqttConnectPacket connectPacket: return EncodeConnectPacket(connectPacket, packetWriter);
  79. case MqttConnAckPacket connAckPacket: return EncodeConnAckPacket(connAckPacket, packetWriter);
  80. case MqttDisconnectPacket _: return EncodeEmptyPacket(MqttControlPacketType.Disconnect);
  81. case MqttPingReqPacket _: return EncodeEmptyPacket(MqttControlPacketType.PingReq);
  82. case MqttPingRespPacket _: return EncodeEmptyPacket(MqttControlPacketType.PingResp);
  83. case MqttPublishPacket publishPacket: return EncodePublishPacket(publishPacket, packetWriter);
  84. case MqttPubAckPacket pubAckPacket: return EncodePubAckPacket(pubAckPacket, packetWriter);
  85. case MqttPubRecPacket pubRecPacket: return EncodePubRecPacket(pubRecPacket, packetWriter);
  86. case MqttPubRelPacket pubRelPacket: return EncodePubRelPacket(pubRelPacket, packetWriter);
  87. case MqttPubCompPacket pubCompPacket: return EncodePubCompPacket(pubCompPacket, packetWriter);
  88. case MqttSubscribePacket subscribePacket: return EncodeSubscribePacket(subscribePacket, packetWriter);
  89. case MqttSubAckPacket subAckPacket: return EncodeSubAckPacket(subAckPacket, packetWriter);
  90. case MqttUnsubscribePacket unsubscribePacket: return EncodeUnsubscribePacket(unsubscribePacket, packetWriter);
  91. case MqttUnsubAckPacket unsubAckPacket: return EncodeUnsubAckPacket(unsubAckPacket, packetWriter);
  92. default: throw new MqttProtocolViolationException("Packet type invalid.");
  93. }
  94. }
  95. private static MqttBasePacket DecodeUnsubAckPacket(IMqttPacketBodyReader body)
  96. {
  97. ThrowIfBodyIsEmpty(body);
  98. return new MqttUnsubAckPacket
  99. {
  100. PacketIdentifier = body.ReadTwoByteInteger()
  101. };
  102. }
  103. private static MqttBasePacket DecodePubCompPacket(IMqttPacketBodyReader body)
  104. {
  105. ThrowIfBodyIsEmpty(body);
  106. return new MqttPubCompPacket
  107. {
  108. PacketIdentifier = body.ReadTwoByteInteger()
  109. };
  110. }
  111. private static MqttBasePacket DecodePubRelPacket(IMqttPacketBodyReader body)
  112. {
  113. ThrowIfBodyIsEmpty(body);
  114. return new MqttPubRelPacket
  115. {
  116. PacketIdentifier = body.ReadTwoByteInteger()
  117. };
  118. }
  119. private static MqttBasePacket DecodePubRecPacket(IMqttPacketBodyReader body)
  120. {
  121. ThrowIfBodyIsEmpty(body);
  122. return new MqttPubRecPacket
  123. {
  124. PacketIdentifier = body.ReadTwoByteInteger()
  125. };
  126. }
  127. private static MqttBasePacket DecodePubAckPacket(IMqttPacketBodyReader body)
  128. {
  129. ThrowIfBodyIsEmpty(body);
  130. return new MqttPubAckPacket
  131. {
  132. PacketIdentifier = body.ReadTwoByteInteger()
  133. };
  134. }
  135. private static MqttBasePacket DecodeUnsubscribePacket(IMqttPacketBodyReader body)
  136. {
  137. ThrowIfBodyIsEmpty(body);
  138. var packet = new MqttUnsubscribePacket
  139. {
  140. PacketIdentifier = body.ReadTwoByteInteger(),
  141. };
  142. while (!body.EndOfStream)
  143. {
  144. packet.TopicFilters.Add(body.ReadStringWithLengthPrefix());
  145. }
  146. return packet;
  147. }
  148. private static MqttBasePacket DecodeSubscribePacket(IMqttPacketBodyReader body)
  149. {
  150. ThrowIfBodyIsEmpty(body);
  151. var packet = new MqttSubscribePacket
  152. {
  153. PacketIdentifier = body.ReadTwoByteInteger()
  154. };
  155. while (!body.EndOfStream)
  156. {
  157. var topicFilter = new MqttTopicFilter
  158. {
  159. Topic = body.ReadStringWithLengthPrefix(),
  160. QualityOfServiceLevel = (MqttQualityOfServiceLevel)body.ReadByte()
  161. };
  162. packet.TopicFilters.Add(topicFilter);
  163. }
  164. return packet;
  165. }
  166. private static MqttBasePacket DecodePublishPacket(ReceivedMqttPacket receivedMqttPacket)
  167. {
  168. ThrowIfBodyIsEmpty(receivedMqttPacket.Body);
  169. var retain = (receivedMqttPacket.FixedHeader & 0x1) > 0;
  170. var qualityOfServiceLevel = (MqttQualityOfServiceLevel)(receivedMqttPacket.FixedHeader >> 1 & 0x3);
  171. var dup = (receivedMqttPacket.FixedHeader & 0x8) > 0;
  172. var topic = receivedMqttPacket.Body.ReadStringWithLengthPrefix();
  173. ushort? packetIdentifier = null;
  174. if (qualityOfServiceLevel > MqttQualityOfServiceLevel.AtMostOnce)
  175. {
  176. packetIdentifier = receivedMqttPacket.Body.ReadTwoByteInteger();
  177. }
  178. var packet = new MqttPublishPacket
  179. {
  180. PacketIdentifier = packetIdentifier,
  181. Retain = retain,
  182. Topic = topic,
  183. QualityOfServiceLevel = qualityOfServiceLevel,
  184. Dup = dup
  185. };
  186. if (!receivedMqttPacket.Body.EndOfStream)
  187. {
  188. packet.Payload = receivedMqttPacket.Body.ReadRemainingData();
  189. }
  190. return packet;
  191. }
  192. private MqttBasePacket DecodeConnectPacket(IMqttPacketBodyReader body)
  193. {
  194. ThrowIfBodyIsEmpty(body);
  195. var protocolName = body.ReadStringWithLengthPrefix();
  196. var protocolVersion = body.ReadByte();
  197. if (protocolName != "MQTT" && protocolName != "MQIsdp")
  198. {
  199. throw new MqttProtocolViolationException("MQTT protocol name do not match MQTT v3.");
  200. }
  201. if (protocolVersion != 3 && protocolVersion != 4)
  202. {
  203. throw new MqttProtocolViolationException("MQTT protocol version do not match MQTT v3.");
  204. }
  205. var packet = new MqttConnectPacket();
  206. var connectFlags = body.ReadByte();
  207. if ((connectFlags & 0x1) > 0)
  208. {
  209. throw new MqttProtocolViolationException("The first bit of the Connect Flags must be set to 0.");
  210. }
  211. packet.CleanSession = (connectFlags & 0x2) > 0;
  212. var willFlag = (connectFlags & 0x4) > 0;
  213. var willQoS = (connectFlags & 0x18) >> 3;
  214. var willRetain = (connectFlags & 0x20) > 0;
  215. var passwordFlag = (connectFlags & 0x40) > 0;
  216. var usernameFlag = (connectFlags & 0x80) > 0;
  217. packet.KeepAlivePeriod = body.ReadTwoByteInteger();
  218. packet.ClientId = body.ReadStringWithLengthPrefix();
  219. if (willFlag)
  220. {
  221. packet.WillMessage = new MqttApplicationMessage
  222. {
  223. Topic = body.ReadStringWithLengthPrefix(),
  224. Payload = body.ReadWithLengthPrefix(),
  225. QualityOfServiceLevel = (MqttQualityOfServiceLevel)willQoS,
  226. Retain = willRetain
  227. };
  228. }
  229. if (usernameFlag)
  230. {
  231. packet.Username = body.ReadStringWithLengthPrefix();
  232. }
  233. if (passwordFlag)
  234. {
  235. packet.Password = body.ReadWithLengthPrefix();
  236. }
  237. ValidateConnectPacket(packet);
  238. return packet;
  239. }
  240. private static MqttBasePacket DecodeSubAckPacket(IMqttPacketBodyReader body)
  241. {
  242. ThrowIfBodyIsEmpty(body);
  243. var packet = new MqttSubAckPacket
  244. {
  245. PacketIdentifier = body.ReadTwoByteInteger()
  246. };
  247. while (!body.EndOfStream)
  248. {
  249. packet.ReturnCodes.Add((MqttSubscribeReturnCode)body.ReadByte());
  250. }
  251. return packet;
  252. }
  253. protected virtual MqttBasePacket DecodeConnAckPacket(IMqttPacketBodyReader body)
  254. {
  255. ThrowIfBodyIsEmpty(body);
  256. var packet = new MqttConnAckPacket();
  257. body.ReadByte(); // Reserved.
  258. packet.ReturnCode = (MqttConnectReturnCode)body.ReadByte();
  259. return packet;
  260. }
  261. protected void ValidateConnectPacket(MqttConnectPacket packet)
  262. {
  263. if (packet == null) throw new ArgumentNullException(nameof(packet));
  264. if (string.IsNullOrEmpty(packet.ClientId) && !packet.CleanSession)
  265. {
  266. throw new MqttProtocolViolationException("CleanSession must be set if ClientId is empty [MQTT-3.1.3-7].");
  267. }
  268. }
  269. // ReSharper disable once ParameterOnlyUsedForPreconditionCheck.Local
  270. private static void ValidatePublishPacket(MqttPublishPacket packet)
  271. {
  272. if (packet.QualityOfServiceLevel == 0 && packet.Dup)
  273. {
  274. throw new MqttProtocolViolationException("Dup flag must be false for QoS 0 packets [MQTT-3.3.1-2].");
  275. }
  276. }
  277. protected virtual byte EncodeConnectPacket(MqttConnectPacket packet, IMqttPacketWriter packetWriter)
  278. {
  279. ValidateConnectPacket(packet);
  280. packetWriter.WriteWithLengthPrefix("MQIsdp");
  281. packetWriter.Write(3); // Protocol Level 3
  282. byte connectFlags = 0x0;
  283. if (packet.CleanSession)
  284. {
  285. connectFlags |= 0x2;
  286. }
  287. if (packet.WillMessage != null)
  288. {
  289. connectFlags |= 0x4;
  290. connectFlags |= (byte)((byte)packet.WillMessage.QualityOfServiceLevel << 3);
  291. if (packet.WillMessage.Retain)
  292. {
  293. connectFlags |= 0x20;
  294. }
  295. }
  296. if (packet.Password != null && packet.Username == null)
  297. {
  298. 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].");
  299. }
  300. if (packet.Password != null)
  301. {
  302. connectFlags |= 0x40;
  303. }
  304. if (packet.Username != null)
  305. {
  306. connectFlags |= 0x80;
  307. }
  308. packetWriter.Write(connectFlags);
  309. packetWriter.Write(packet.KeepAlivePeriod);
  310. packetWriter.WriteWithLengthPrefix(packet.ClientId);
  311. if (packet.WillMessage != null)
  312. {
  313. packetWriter.WriteWithLengthPrefix(packet.WillMessage.Topic);
  314. packetWriter.WriteWithLengthPrefix(packet.WillMessage.Payload);
  315. }
  316. if (packet.Username != null)
  317. {
  318. packetWriter.WriteWithLengthPrefix(packet.Username);
  319. }
  320. if (packet.Password != null)
  321. {
  322. packetWriter.WriteWithLengthPrefix(packet.Password);
  323. }
  324. return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.Connect);
  325. }
  326. protected virtual byte EncodeConnAckPacket(MqttConnAckPacket packet, IMqttPacketWriter packetWriter)
  327. {
  328. packetWriter.Write(0); // Reserved.
  329. packetWriter.Write((byte)packet.ReturnCode.Value);
  330. return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.ConnAck);
  331. }
  332. private static byte EncodePubRelPacket(MqttPubRelPacket packet, IMqttPacketWriter packetWriter)
  333. {
  334. if (!packet.PacketIdentifier.HasValue)
  335. {
  336. throw new MqttProtocolViolationException("PubRel packet has no packet identifier.");
  337. }
  338. packetWriter.Write(packet.PacketIdentifier.Value);
  339. return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.PubRel, 0x02);
  340. }
  341. private static byte EncodePublishPacket(MqttPublishPacket packet, IMqttPacketWriter packetWriter)
  342. {
  343. ValidatePublishPacket(packet);
  344. packetWriter.WriteWithLengthPrefix(packet.Topic);
  345. if (packet.QualityOfServiceLevel > MqttQualityOfServiceLevel.AtMostOnce)
  346. {
  347. if (!packet.PacketIdentifier.HasValue)
  348. {
  349. throw new MqttProtocolViolationException("Publish packet has no packet identifier.");
  350. }
  351. packetWriter.Write(packet.PacketIdentifier.Value);
  352. }
  353. else
  354. {
  355. if (packet.PacketIdentifier > 0)
  356. {
  357. throw new MqttProtocolViolationException("Packet identifier must be empty if QoS == 0 [MQTT-2.3.1-5].");
  358. }
  359. }
  360. if (packet.Payload?.Length > 0)
  361. {
  362. packetWriter.Write(packet.Payload, 0, packet.Payload.Length);
  363. }
  364. byte fixedHeader = 0;
  365. if (packet.Retain)
  366. {
  367. fixedHeader |= 0x01;
  368. }
  369. fixedHeader |= (byte)((byte)packet.QualityOfServiceLevel << 1);
  370. if (packet.Dup)
  371. {
  372. fixedHeader |= 0x08;
  373. }
  374. return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.Publish, fixedHeader);
  375. }
  376. private static byte EncodePubAckPacket(MqttPubAckPacket packet, IMqttPacketWriter packetWriter)
  377. {
  378. if (!packet.PacketIdentifier.HasValue)
  379. {
  380. throw new MqttProtocolViolationException("PubAck packet has no packet identifier.");
  381. }
  382. packetWriter.Write(packet.PacketIdentifier.Value);
  383. return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.PubAck);
  384. }
  385. private static byte EncodePubRecPacket(MqttPubRecPacket packet, IMqttPacketWriter packetWriter)
  386. {
  387. if (!packet.PacketIdentifier.HasValue)
  388. {
  389. throw new MqttProtocolViolationException("PubRec packet has no packet identifier.");
  390. }
  391. packetWriter.Write(packet.PacketIdentifier.Value);
  392. return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.PubRec);
  393. }
  394. private static byte EncodePubCompPacket(MqttPubCompPacket packet, IMqttPacketWriter packetWriter)
  395. {
  396. if (!packet.PacketIdentifier.HasValue)
  397. {
  398. throw new MqttProtocolViolationException("PubComp packet has no packet identifier.");
  399. }
  400. packetWriter.Write(packet.PacketIdentifier.Value);
  401. return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.PubComp);
  402. }
  403. private static byte EncodeSubscribePacket(MqttSubscribePacket packet, IMqttPacketWriter packetWriter)
  404. {
  405. if (!packet.TopicFilters.Any()) throw new MqttProtocolViolationException("At least one topic filter must be set [MQTT-3.8.3-3].");
  406. if (!packet.PacketIdentifier.HasValue)
  407. {
  408. throw new MqttProtocolViolationException("Subscribe packet has no packet identifier.");
  409. }
  410. packetWriter.Write(packet.PacketIdentifier.Value);
  411. if (packet.TopicFilters?.Count > 0)
  412. {
  413. foreach (var topicFilter in packet.TopicFilters)
  414. {
  415. packetWriter.WriteWithLengthPrefix(topicFilter.Topic);
  416. packetWriter.Write((byte)topicFilter.QualityOfServiceLevel);
  417. }
  418. }
  419. return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.Subscribe, 0x02);
  420. }
  421. private static byte EncodeSubAckPacket(MqttSubAckPacket packet, IMqttPacketWriter packetWriter)
  422. {
  423. if (!packet.PacketIdentifier.HasValue)
  424. {
  425. throw new MqttProtocolViolationException("SubAck packet has no packet identifier.");
  426. }
  427. packetWriter.Write(packet.PacketIdentifier.Value);
  428. if (packet.ReturnCodes?.Any() == true)
  429. {
  430. foreach (var packetSubscribeReturnCode in packet.ReturnCodes)
  431. {
  432. packetWriter.Write((byte)packetSubscribeReturnCode);
  433. }
  434. }
  435. return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.SubAck);
  436. }
  437. private static byte EncodeUnsubscribePacket(MqttUnsubscribePacket packet, IMqttPacketWriter packetWriter)
  438. {
  439. if (!packet.TopicFilters.Any()) throw new MqttProtocolViolationException("At least one topic filter must be set [MQTT-3.10.3-2].");
  440. if (!packet.PacketIdentifier.HasValue)
  441. {
  442. throw new MqttProtocolViolationException("Unsubscribe packet has no packet identifier.");
  443. }
  444. packetWriter.Write(packet.PacketIdentifier.Value);
  445. if (packet.TopicFilters?.Any() == true)
  446. {
  447. foreach (var topicFilter in packet.TopicFilters)
  448. {
  449. packetWriter.WriteWithLengthPrefix(topicFilter);
  450. }
  451. }
  452. return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.Unsubscibe, 0x02);
  453. }
  454. private static byte EncodeUnsubAckPacket(MqttUnsubAckPacket packet, IMqttPacketWriter packetWriter)
  455. {
  456. if (!packet.PacketIdentifier.HasValue)
  457. {
  458. throw new MqttProtocolViolationException("UnsubAck packet has no packet identifier.");
  459. }
  460. packetWriter.Write(packet.PacketIdentifier.Value);
  461. return MqttPacketWriter.BuildFixedHeader(MqttControlPacketType.UnsubAck);
  462. }
  463. private static byte EncodeEmptyPacket(MqttControlPacketType type)
  464. {
  465. return MqttPacketWriter.BuildFixedHeader(type);
  466. }
  467. // ReSharper disable once ParameterOnlyUsedForPreconditionCheck.Local
  468. protected static void ThrowIfBodyIsEmpty(IMqttPacketBodyReader body)
  469. {
  470. if (body == null || body.Length == 0)
  471. {
  472. throw new MqttProtocolViolationException("Data from the body is required but not present.");
  473. }
  474. }
  475. }
  476. }