3
0

header.go 4.9 KB

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