/org.eclipse.paho.mqttv5.common/src/main/java/org/eclipse/paho/mqttv5/common/packet/MqttDataTypes.java

http://github.com/eclipse/paho.mqtt.java · Java · 253 lines · 156 code · 26 blank · 71 comment · 40 complexity · bb2217f39f1ee178a6a14790b17b4b54 MD5 · raw file

  1. package org.eclipse.paho.mqttv5.common.packet;
  2. import java.io.ByteArrayOutputStream;
  3. import java.io.DataInputStream;
  4. import java.io.DataOutputStream;
  5. import java.io.EOFException;
  6. import java.io.IOException;
  7. import java.nio.charset.Charset;
  8. import java.nio.charset.StandardCharsets;
  9. import org.eclipse.paho.mqttv5.common.MqttException;
  10. import org.eclipse.paho.mqttv5.common.packet.util.VariableByteInteger;
  11. public class MqttDataTypes {
  12. private static final int TWO_BYTE_INT_MAX = 65535;
  13. private static final long FOUR_BYTE_INT_MAX = 4294967295L;
  14. public static final int VARIABLE_BYTE_INT_MAX = 268435455;
  15. public MqttDataTypes() throws IllegalAccessException {
  16. throw new IllegalAccessException("Utility Class");
  17. }
  18. public static void validateTwoByteInt(Integer value) throws IllegalArgumentException {
  19. if (value == null) {
  20. return;
  21. }
  22. if (value >= 0 && value <= TWO_BYTE_INT_MAX) {
  23. return;
  24. } else {
  25. throw new IllegalArgumentException("This property must be a number between 0 and " + TWO_BYTE_INT_MAX);
  26. }
  27. }
  28. public static void validateFourByteInt(Long value) throws IllegalArgumentException {
  29. if (value == null) {
  30. return;
  31. }
  32. if (value >= 0 && value <= FOUR_BYTE_INT_MAX) {
  33. return;
  34. } else {
  35. throw new IllegalArgumentException("This property must be a number between 0 and " + FOUR_BYTE_INT_MAX);
  36. }
  37. }
  38. public static void validateVariableByteInt(int value) throws IllegalArgumentException {
  39. if (value >= 0 && value <= VARIABLE_BYTE_INT_MAX) {
  40. return;
  41. } else {
  42. throw new IllegalArgumentException("This property must be a number between 0 and " + VARIABLE_BYTE_INT_MAX);
  43. }
  44. }
  45. public static void writeUnsignedFourByteInt(long value, DataOutputStream stream) throws IOException {
  46. stream.writeByte((byte) (value >>> 24));
  47. stream.writeByte((byte) (value >>> 16));
  48. stream.writeByte((byte) (value >>> 8));
  49. stream.writeByte((byte) (value >>> 0));
  50. }
  51. /**
  52. * Reads a Four Byte Integer, then converts it to a float. This is because Java
  53. * doesn't have Unsigned Integers.
  54. *
  55. * @param inputStream
  56. * The input stream to read from.
  57. * @return a {@link Long} containing the value of the Four Byte int (Between 0
  58. * and 4294967295)
  59. * @throws IOException
  60. * if an exception occurs whilst reading from the Input Stream
  61. */
  62. public static Long readUnsignedFourByteInt(DataInputStream inputStream) throws IOException {
  63. byte[] readBuffer = {0, 0, 0, 0, 0, 0, 0, 0};
  64. inputStream.readFully(readBuffer, 4, 4);
  65. return (((long) readBuffer[0] << 56) + ((long) (readBuffer[1] & 255) << 48)
  66. + ((long) (readBuffer[2] & 255) << 40) + ((long) (readBuffer[3] & 255) << 32)
  67. + ((long) (readBuffer[4] & 255) << 24) + ((readBuffer[5] & 255) << 16) + ((readBuffer[6] & 255) << 8)
  68. + ((readBuffer[7] & 255) << 0));
  69. }
  70. /**
  71. * Reads a Two Byte Integer, this is because Java does not have unsigned
  72. * integers.
  73. *
  74. * @param inputStream
  75. * The input stream to read from.
  76. * @return a {@link int} containing the value of the Two Byte int (Between 0 and
  77. * 65535)
  78. * @throws IOException
  79. * if an exception occurs whilst reading from the Input Stream
  80. *
  81. */
  82. public static int readUnsignedTwoByteInt(DataInputStream inputStream) throws IOException {
  83. // byte readBuffer[] = {0,0}
  84. int ch1 = inputStream.read();
  85. int ch2 = inputStream.read();
  86. if ((ch1 | ch2) < 0)
  87. throw new EOFException();
  88. return (int) ((ch1 << 8) + (ch2 << 0));
  89. }
  90. /**
  91. * Encodes a String given into UTF-8, before writing this to the
  92. * {@link DataOutputStream} the length of the encoded string is encoded into two
  93. * bytes and then written to the {@link DataOutputStream}.
  94. * {@link DataOutputStream#writeUTF(String)} should be no longer used.
  95. * {@link DataOutputStream#writeUTF(String)} does not correctly encode UTF-16
  96. * surrogate characters.
  97. *
  98. * @param dos
  99. * The stream to write the encoded UTF-8 string to.
  100. * @param stringToEncode
  101. * The string to be encoded
  102. * @throws MqttException
  103. * Thrown when an error occurs with either the encoding or writing
  104. * the data to the stream.
  105. */
  106. public static void encodeUTF8(DataOutputStream dos, String stringToEncode) throws MqttException {
  107. validateUTF8String(stringToEncode);
  108. try {
  109. byte[] encodedString = stringToEncode.getBytes(STRING_ENCODING);
  110. byte byte1 = (byte) ((encodedString.length >>> 8) & 0xFF);
  111. byte byte2 = (byte) ((encodedString.length >>> 0) & 0xFF);
  112. dos.write(byte1);
  113. dos.write(byte2);
  114. dos.write(encodedString);
  115. } catch (IOException ex) {
  116. throw new MqttException(ex);
  117. }
  118. }
  119. protected static final Charset STRING_ENCODING = StandardCharsets.UTF_8;
  120. /**
  121. * Decodes a UTF-8 string from the {@link DataInputStream} provided.
  122. * {@link DataInputStream#readUTF()} should be no longer used, because
  123. * {@link DataInputStream#readUTF()} does not decode UTF-16 surrogate characters
  124. * correctly.
  125. *
  126. * @param input
  127. * The input stream from which to read the encoded string.
  128. * @return a decoded String from the {@link DataInputStream}.
  129. * @throws MqttException
  130. * thrown when an error occurs with either reading from the stream
  131. * or decoding the encoding string.
  132. */
  133. public static String decodeUTF8(DataInputStream input) throws MqttException {
  134. int encodedLength;
  135. try {
  136. encodedLength = input.readUnsignedShort();
  137. byte[] encodedString = new byte[encodedLength];
  138. input.readFully(encodedString);
  139. String output = new String(encodedString, STRING_ENCODING);
  140. validateUTF8String(output);
  141. return output;
  142. } catch (IOException ioe) {
  143. throw new MqttException(MqttException.REASON_CODE_MALFORMED_PACKET, ioe);
  144. }
  145. }
  146. /**
  147. * Validate a UTF-8 String for suitability for MQTT.
  148. *
  149. * @param input
  150. * - The Input String
  151. * @throws IllegalArgumentException
  152. */
  153. private static void validateUTF8String(String input) throws IllegalArgumentException {
  154. for (int i = 0; i < input.length(); i++) {
  155. boolean isBad = false;
  156. char c = input.charAt(i);
  157. /* Check for mismatched surrogates */
  158. if (Character.isHighSurrogate(c)) {
  159. if (++i == input.length()) {
  160. isBad = true; /* Trailing high surrogate */
  161. } else {
  162. char c2 = input.charAt(i);
  163. if (!Character.isLowSurrogate(c2)) {
  164. isBad = true; /* No low surrogate */
  165. } else {
  166. int ch = ((((int) c) & 0x3ff) << 10) | (c2 & 0x3ff);
  167. if ((ch & 0xffff) == 0xffff || (ch & 0xffff) == 0xfffe) {
  168. isBad = true; /* Noncharacter in base plane */
  169. }
  170. }
  171. }
  172. } else {
  173. if (Character.isISOControl(c) || Character.isLowSurrogate(c)) {
  174. isBad = true; /* Control character or no high surrogate */
  175. } else if (c >= 0xfdd0 && (c <= 0xfddf || c >= 0xfffe)) {
  176. isBad = true; /* Noncharacter in other nonbase plane */
  177. }
  178. }
  179. if (isBad) {
  180. throw new IllegalArgumentException(String.format("Invalid UTF-8 char: [%04x]", (int) c));
  181. }
  182. }
  183. }
  184. /**
  185. * Decodes an MQTT Multi-Byte Integer from the given stream
  186. *
  187. * @param in
  188. * the DataInputStream to decode a Variable Byte Integer From
  189. * @return a new VariableByteInteger
  190. * @throws IOException
  191. * if an error occurred whilst decoding the VBI
  192. */
  193. public static VariableByteInteger readVariableByteInteger(DataInputStream in) throws IOException {
  194. byte digit;
  195. int value = 0;
  196. int multiplier = 1;
  197. int count = 0;
  198. do {
  199. digit = in.readByte();
  200. count++;
  201. value += ((digit & 0x7F) * multiplier);
  202. multiplier *= 128;
  203. } while ((digit & 0x80) != 0);
  204. if (value < 0 || value > VARIABLE_BYTE_INT_MAX) {
  205. throw new IOException("This property must be a number between 0 and " + VARIABLE_BYTE_INT_MAX
  206. + ". Read value was: " + value);
  207. }
  208. return new VariableByteInteger(value, count);
  209. }
  210. public static byte[] encodeVariableByteInteger(int number) throws IllegalArgumentException {
  211. validateVariableByteInt(number);
  212. int numBytes = 0;
  213. long no = number;
  214. ByteArrayOutputStream baos = new ByteArrayOutputStream();
  215. // Encode the remaining length fields in the four bytes
  216. do {
  217. byte digit = (byte) (no % 128);
  218. no = no / 128;
  219. if (no > 0) {
  220. digit |= 0x80;
  221. }
  222. baos.write(digit);
  223. numBytes++;
  224. } while ((no > 0) && (numBytes < 4));
  225. return baos.toByteArray();
  226. }
  227. }