/src/java/org/apache/cassandra/net/MessageOut.java

http://github.com/apache/cassandra · Java · 153 lines · 114 code · 21 blank · 18 comment · 8 complexity · fcc5b493d9e27000c4a0935f5095fe43 MD5 · raw file

  1. /**
  2. * Licensed to the Apache Software Foundation (ASF) under one
  3. * or more contributor license agreements. See the NOTICE file
  4. * distributed with this work for additional information
  5. * regarding copyright ownership. The ASF licenses this file
  6. * to you under the Apache License, Version 2.0 (the
  7. * "License"); you may not use this file except in compliance
  8. * with the License. You may obtain a copy of the License at
  9. *
  10. * http://www.apache.org/licenses/LICENSE-2.0
  11. *
  12. * Unless required by applicable law or agreed to in writing, software
  13. * distributed under the License is distributed on an "AS IS" BASIS,
  14. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  15. * See the License for the specific language governing permissions and
  16. * limitations under the License.
  17. */
  18. package org.apache.cassandra.net;
  19. import java.io.DataOutputStream;
  20. import java.io.IOException;
  21. import java.net.InetAddress;
  22. import java.util.Collections;
  23. import java.util.Map;
  24. import com.google.common.annotations.VisibleForTesting;
  25. import com.google.common.collect.ImmutableMap;
  26. import org.apache.cassandra.concurrent.Stage;
  27. import org.apache.cassandra.config.DatabaseDescriptor;
  28. import org.apache.cassandra.db.TypeSizes;
  29. import org.apache.cassandra.io.IVersionedSerializer;
  30. import org.apache.cassandra.tracing.Tracing;
  31. import org.apache.cassandra.utils.FBUtilities;
  32. import org.apache.cassandra.utils.UUIDGen;
  33. import static org.apache.cassandra.tracing.Tracing.TRACE_HEADER;
  34. import static org.apache.cassandra.tracing.Tracing.isTracing;
  35. public class MessageOut<T>
  36. {
  37. public final InetAddress from;
  38. public final MessagingService.Verb verb;
  39. public final T payload;
  40. public final IVersionedSerializer<T> serializer;
  41. public final Map<String, byte[]> parameters;
  42. // we do support messages that just consist of a verb
  43. public MessageOut(MessagingService.Verb verb)
  44. {
  45. this(verb, null, null);
  46. }
  47. public MessageOut(MessagingService.Verb verb, T payload, IVersionedSerializer<T> serializer)
  48. {
  49. this(verb,
  50. payload,
  51. serializer,
  52. isTracing() ? ImmutableMap.of(TRACE_HEADER, UUIDGen.decompose(Tracing.instance().getSessionId()))
  53. : Collections.<String, byte[]>emptyMap());
  54. }
  55. private MessageOut(MessagingService.Verb verb, T payload, IVersionedSerializer<T> serializer, Map<String, byte[]> parameters)
  56. {
  57. this(FBUtilities.getBroadcastAddress(), verb, payload, serializer, parameters);
  58. }
  59. @VisibleForTesting
  60. public MessageOut(InetAddress from, MessagingService.Verb verb, T payload, IVersionedSerializer<T> serializer, Map<String, byte[]> parameters)
  61. {
  62. this.from = from;
  63. this.verb = verb;
  64. this.payload = payload;
  65. this.serializer = serializer;
  66. this.parameters = parameters;
  67. }
  68. public MessageOut<T> withParameter(String key, byte[] value)
  69. {
  70. ImmutableMap.Builder<String, byte[]> builder = ImmutableMap.builder();
  71. builder.putAll(parameters).put(key, value);
  72. return new MessageOut<T>(verb, payload, serializer, builder.build());
  73. }
  74. public MessageOut withHeaderRemoved(String key)
  75. {
  76. ImmutableMap.Builder<String, byte[]> builder = ImmutableMap.builder();
  77. for (Map.Entry<String, byte[]> entry : parameters.entrySet())
  78. {
  79. if (!entry.getKey().equals(key))
  80. builder.put(entry.getKey(), entry.getValue());
  81. }
  82. return new MessageOut<T>(verb, payload, serializer, builder.build());
  83. }
  84. public Stage getStage()
  85. {
  86. return MessagingService.verbStages.get(verb);
  87. }
  88. public long getTimeout()
  89. {
  90. return DatabaseDescriptor.getTimeout(verb);
  91. }
  92. public String toString()
  93. {
  94. StringBuilder sbuf = new StringBuilder("");
  95. sbuf.append("TYPE:").append(getStage()).append(" VERB:").append(verb);
  96. return sbuf.toString();
  97. }
  98. public void serialize(DataOutputStream out, int version) throws IOException
  99. {
  100. CompactEndpointSerializationHelper.serialize(from, out);
  101. out.writeInt(verb.ordinal());
  102. out.writeInt(parameters.size());
  103. for (Map.Entry<String, byte[]> entry : parameters.entrySet())
  104. {
  105. out.writeUTF(entry.getKey());
  106. out.writeInt(entry.getValue().length);
  107. out.write(entry.getValue());
  108. }
  109. long longSize = payload == null ? 0 : serializer.serializedSize(payload, version);
  110. assert longSize <= Integer.MAX_VALUE; // larger values are supported in sstables but not messages
  111. out.writeInt((int) longSize);
  112. if (payload != null)
  113. serializer.serialize(payload, out, version);
  114. }
  115. public int serializedSize(int version)
  116. {
  117. int size = CompactEndpointSerializationHelper.serializedSize(from);
  118. size += TypeSizes.NATIVE.sizeof(verb.ordinal());
  119. size += TypeSizes.NATIVE.sizeof(parameters.size());
  120. for (Map.Entry<String, byte[]> entry : parameters.entrySet())
  121. {
  122. TypeSizes.NATIVE.sizeof(entry.getKey());
  123. TypeSizes.NATIVE.sizeof(entry.getValue().length);
  124. size += entry.getValue().length;
  125. }
  126. long longSize = payload == null ? 0 : serializer.serializedSize(payload, version);
  127. assert longSize <= Integer.MAX_VALUE; // larger values are supported in sstables but not messages
  128. size += TypeSizes.NATIVE.sizeof((int) longSize);
  129. size += longSize;
  130. return size;
  131. }
  132. }