2
0

header.go 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. package header
  2. import (
  3. "encoding/binary"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. )
  8. //Version 1 header:
  9. // 0 31
  10. // |-----------------------------------------------------------------------|
  11. // | Version (uint4) | Type (uint4) | Subtype (uint8) | Reserved (uint16) | 32
  12. // |-----------------------------------------------------------------------|
  13. // | Remote index (uint32) | 64
  14. // |-----------------------------------------------------------------------|
  15. // | Message counter | 96
  16. // | (uint64) | 128
  17. // |-----------------------------------------------------------------------|
  18. // | payload... |
  19. type m map[string]interface{}
  20. const (
  21. Version uint8 = 1
  22. Len = 16
  23. )
  24. type MessageType uint8
  25. type MessageSubType uint8
  26. const (
  27. Handshake MessageType = 0
  28. Message MessageType = 1
  29. RecvError MessageType = 2
  30. LightHouse MessageType = 3
  31. Test MessageType = 4
  32. CloseTunnel MessageType = 5
  33. Control MessageType = 6
  34. )
  35. var typeMap = map[MessageType]string{
  36. Handshake: "handshake",
  37. Message: "message",
  38. RecvError: "recvError",
  39. LightHouse: "lightHouse",
  40. Test: "test",
  41. CloseTunnel: "closeTunnel",
  42. Control: "control",
  43. }
  44. const (
  45. MessageNone MessageSubType = 0
  46. MessageRelay MessageSubType = 1
  47. )
  48. const (
  49. TestRequest MessageSubType = 0
  50. TestReply MessageSubType = 1
  51. )
  52. const (
  53. HandshakeIXPSK0 MessageSubType = 0
  54. HandshakeXXPSK0 MessageSubType = 1
  55. )
  56. var ErrHeaderTooShort = errors.New("header is too short")
  57. var subTypeTestMap = map[MessageSubType]string{
  58. TestRequest: "testRequest",
  59. TestReply: "testReply",
  60. }
  61. var subTypeNoneMap = map[MessageSubType]string{0: "none"}
  62. var subTypeMap = map[MessageType]*map[MessageSubType]string{
  63. Message: {
  64. MessageNone: "none",
  65. MessageRelay: "relay",
  66. },
  67. RecvError: &subTypeNoneMap,
  68. LightHouse: &subTypeNoneMap,
  69. Test: &subTypeTestMap,
  70. CloseTunnel: &subTypeNoneMap,
  71. Handshake: {
  72. HandshakeIXPSK0: "ix_psk0",
  73. },
  74. Control: &subTypeNoneMap,
  75. }
  76. type H struct {
  77. Version uint8
  78. Type MessageType
  79. Subtype MessageSubType
  80. Reserved uint16
  81. RemoteIndex uint32
  82. MessageCounter uint64
  83. }
  84. // Encode uses the provided byte array to encode the provided header values into.
  85. // Byte array must be capped higher than HeaderLen or this will panic
  86. func Encode(b []byte, v uint8, t MessageType, st MessageSubType, ri uint32, c uint64) []byte {
  87. b = b[:Len]
  88. b[0] = v<<4 | byte(t&0x0f)
  89. b[1] = byte(st)
  90. binary.BigEndian.PutUint16(b[2:4], 0)
  91. binary.BigEndian.PutUint32(b[4:8], ri)
  92. binary.BigEndian.PutUint64(b[8:16], c)
  93. return b
  94. }
  95. // String creates a readable string representation of a header
  96. func (h *H) String() string {
  97. if h == nil {
  98. return "<nil>"
  99. }
  100. return fmt.Sprintf("ver=%d type=%s subtype=%s reserved=%#x remoteindex=%v messagecounter=%v",
  101. h.Version, h.TypeName(), h.SubTypeName(), h.Reserved, h.RemoteIndex, h.MessageCounter)
  102. }
  103. // MarshalJSON creates a json string representation of a header
  104. func (h *H) MarshalJSON() ([]byte, error) {
  105. return json.Marshal(m{
  106. "version": h.Version,
  107. "type": h.TypeName(),
  108. "subType": h.SubTypeName(),
  109. "reserved": h.Reserved,
  110. "remoteIndex": h.RemoteIndex,
  111. "messageCounter": h.MessageCounter,
  112. })
  113. }
  114. // Encode turns header into bytes
  115. func (h *H) Encode(b []byte) ([]byte, error) {
  116. if h == nil {
  117. return nil, errors.New("nil header")
  118. }
  119. return Encode(b, h.Version, h.Type, h.Subtype, h.RemoteIndex, h.MessageCounter), nil
  120. }
  121. // Parse is a helper function to parses given bytes into new Header struct
  122. func (h *H) Parse(b []byte) error {
  123. if len(b) < Len {
  124. return ErrHeaderTooShort
  125. }
  126. // get upper 4 bytes
  127. h.Version = uint8((b[0] >> 4) & 0x0f)
  128. // get lower 4 bytes
  129. h.Type = MessageType(b[0] & 0x0f)
  130. h.Subtype = MessageSubType(b[1])
  131. h.Reserved = binary.BigEndian.Uint16(b[2:4])
  132. h.RemoteIndex = binary.BigEndian.Uint32(b[4:8])
  133. h.MessageCounter = binary.BigEndian.Uint64(b[8:16])
  134. return nil
  135. }
  136. // TypeName will transform the headers message type into a human string
  137. func (h *H) TypeName() string {
  138. return TypeName(h.Type)
  139. }
  140. // TypeName will transform a nebula message type into a human string
  141. func TypeName(t MessageType) string {
  142. if n, ok := typeMap[t]; ok {
  143. return n
  144. }
  145. return "unknown"
  146. }
  147. // SubTypeName will transform the headers message sub type into a human string
  148. func (h *H) SubTypeName() string {
  149. return SubTypeName(h.Type, h.Subtype)
  150. }
  151. // SubTypeName will transform a nebula message sub type into a human string
  152. func SubTypeName(t MessageType, s MessageSubType) string {
  153. if n, ok := subTypeMap[t]; ok {
  154. if x, ok := (*n)[s]; ok {
  155. return x
  156. }
  157. }
  158. return "unknown"
  159. }
  160. // NewHeader turns bytes into a header
  161. func NewHeader(b []byte) (*H, error) {
  162. h := new(H)
  163. if err := h.Parse(b); err != nil {
  164. return nil, err
  165. }
  166. return h, nil
  167. }