header.go 4.8 KB

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